django-cfg 1.3.1__py3-none-any.whl → 1.3.5__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 (115) hide show
  1. django_cfg/__init__.py +1 -1
  2. django_cfg/apps/payments/admin_interface/old/payments/base.html +175 -0
  3. django_cfg/apps/payments/admin_interface/old/payments/components/dev_tool_card.html +125 -0
  4. django_cfg/apps/payments/admin_interface/old/payments/components/ngrok_status_card.html +113 -0
  5. django_cfg/apps/payments/admin_interface/old/payments/components/status_card.html +35 -0
  6. django_cfg/apps/payments/admin_interface/old/payments/payment_dashboard.html +309 -0
  7. django_cfg/apps/payments/admin_interface/old/payments/payment_form.html +303 -0
  8. django_cfg/apps/payments/admin_interface/old/payments/payment_list.html +382 -0
  9. django_cfg/apps/payments/admin_interface/old/payments/webhook_dashboard.html +518 -0
  10. django_cfg/apps/payments/{static → admin_interface/old/static}/payments/css/components.css +248 -9
  11. django_cfg/apps/payments/admin_interface/old/static/payments/js/ngrok-status.js +163 -0
  12. django_cfg/apps/payments/admin_interface/serializers/__init__.py +39 -0
  13. django_cfg/apps/payments/admin_interface/serializers/payment_serializers.py +149 -0
  14. django_cfg/apps/payments/admin_interface/serializers/webhook_serializers.py +114 -0
  15. django_cfg/apps/payments/admin_interface/templates/payments/base.html +55 -90
  16. django_cfg/apps/payments/admin_interface/templates/payments/components/dialog.html +81 -0
  17. django_cfg/apps/payments/admin_interface/templates/payments/components/ngrok_help_dialog.html +112 -0
  18. django_cfg/apps/payments/admin_interface/templates/payments/components/ngrok_status.html +175 -0
  19. django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +21 -17
  20. django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +123 -250
  21. django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +170 -269
  22. django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +152 -355
  23. django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +202 -551
  24. django_cfg/apps/payments/admin_interface/views/__init__.py +25 -14
  25. django_cfg/apps/payments/admin_interface/views/api/__init__.py +20 -0
  26. django_cfg/apps/payments/admin_interface/views/api/payments.py +191 -0
  27. django_cfg/apps/payments/admin_interface/views/api/stats.py +206 -0
  28. django_cfg/apps/payments/admin_interface/views/api/users.py +60 -0
  29. django_cfg/apps/payments/admin_interface/views/api/webhook_admin.py +257 -0
  30. django_cfg/apps/payments/admin_interface/views/api/webhook_public.py +70 -0
  31. django_cfg/apps/payments/admin_interface/views/base.py +114 -0
  32. django_cfg/apps/payments/admin_interface/views/dashboard.py +60 -0
  33. django_cfg/apps/payments/admin_interface/views/forms.py +94 -0
  34. django_cfg/apps/payments/config/helpers.py +2 -2
  35. django_cfg/apps/payments/management/commands/cleanup_expired_data.py +429 -0
  36. django_cfg/apps/payments/management/commands/currency_stats.py +443 -0
  37. django_cfg/apps/payments/management/commands/manage_currencies.py +9 -20
  38. django_cfg/apps/payments/management/commands/manage_providers.py +5 -5
  39. django_cfg/apps/payments/management/commands/process_pending_payments.py +357 -0
  40. django_cfg/apps/payments/management/commands/test_providers.py +434 -0
  41. django_cfg/apps/payments/middleware/api_access.py +35 -34
  42. django_cfg/apps/payments/migrations/0001_initial.py +1 -1
  43. django_cfg/apps/payments/models/balance.py +5 -2
  44. django_cfg/apps/payments/models/managers/api_key_managers.py +6 -2
  45. django_cfg/apps/payments/models/managers/balance_managers.py +3 -3
  46. django_cfg/apps/payments/models/managers/payment_managers.py +5 -0
  47. django_cfg/apps/payments/models/managers/subscription_managers.py +3 -3
  48. django_cfg/apps/payments/models/subscriptions.py +0 -24
  49. django_cfg/apps/payments/services/cache/__init__.py +1 -1
  50. django_cfg/apps/payments/services/cache_service/__init__.py +143 -0
  51. django_cfg/apps/payments/services/cache_service/api_key_cache.py +37 -0
  52. django_cfg/apps/payments/services/cache_service/interfaces.py +32 -0
  53. django_cfg/apps/payments/services/cache_service/keys.py +49 -0
  54. django_cfg/apps/payments/services/cache_service/rate_limit_cache.py +47 -0
  55. django_cfg/apps/payments/services/cache_service/simple_cache.py +101 -0
  56. django_cfg/apps/payments/services/core/balance_service.py +13 -2
  57. django_cfg/apps/payments/services/core/payment_service.py +49 -22
  58. django_cfg/apps/payments/services/integrations/ngrok_service.py +3 -3
  59. django_cfg/apps/payments/services/providers/registry.py +20 -0
  60. django_cfg/apps/payments/signals/api_key_signals.py +2 -2
  61. django_cfg/apps/payments/signals/balance_signals.py +8 -5
  62. django_cfg/apps/payments/static/payments/js/api-client.js +385 -0
  63. django_cfg/apps/payments/static/payments/js/ngrok-status.js +58 -0
  64. django_cfg/apps/payments/static/payments/js/payment-dashboard.js +50 -0
  65. django_cfg/apps/payments/static/payments/js/payment-form.js +175 -0
  66. django_cfg/apps/payments/static/payments/js/payment-list.js +95 -0
  67. django_cfg/apps/payments/static/payments/js/webhook-dashboard.js +154 -0
  68. django_cfg/apps/payments/urls.py +4 -0
  69. django_cfg/apps/payments/urls_admin.py +37 -18
  70. django_cfg/apps/payments/views/api/api_keys.py +14 -0
  71. django_cfg/apps/payments/views/api/base.py +1 -0
  72. django_cfg/apps/payments/views/api/currencies.py +2 -2
  73. django_cfg/apps/payments/views/api/payments.py +11 -5
  74. django_cfg/apps/payments/views/api/subscriptions.py +36 -31
  75. django_cfg/apps/payments/views/overview/__init__.py +40 -0
  76. django_cfg/apps/payments/views/overview/serializers.py +205 -0
  77. django_cfg/apps/payments/views/overview/services.py +439 -0
  78. django_cfg/apps/payments/views/overview/urls.py +27 -0
  79. django_cfg/apps/payments/views/overview/views.py +231 -0
  80. django_cfg/apps/payments/views/serializers/api_keys.py +20 -6
  81. django_cfg/apps/payments/views/serializers/balances.py +5 -8
  82. django_cfg/apps/payments/views/serializers/currencies.py +2 -6
  83. django_cfg/apps/payments/views/serializers/payments.py +37 -32
  84. django_cfg/apps/payments/views/serializers/subscriptions.py +4 -26
  85. django_cfg/apps/urls.py +2 -1
  86. django_cfg/core/config.py +25 -15
  87. django_cfg/core/generation.py +12 -12
  88. django_cfg/core/integration/display/startup.py +1 -1
  89. django_cfg/core/validation.py +4 -4
  90. django_cfg/management/commands/show_config.py +2 -2
  91. django_cfg/management/commands/tree.py +1 -3
  92. django_cfg/middleware/__init__.py +2 -0
  93. django_cfg/middleware/static_nocache.py +55 -0
  94. django_cfg/models/payments.py +13 -15
  95. django_cfg/models/security.py +15 -0
  96. django_cfg/modules/django_ngrok.py +6 -0
  97. django_cfg/modules/django_unfold/dashboard.py +1 -3
  98. django_cfg/utils/smart_defaults.py +51 -5
  99. {django_cfg-1.3.1.dist-info → django_cfg-1.3.5.dist-info}/METADATA +1 -1
  100. {django_cfg-1.3.1.dist-info → django_cfg-1.3.5.dist-info}/RECORD +111 -69
  101. django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +0 -38
  102. django_cfg/apps/payments/admin_interface/views/payment_views.py +0 -259
  103. django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +0 -37
  104. django_cfg/apps/payments/services/cache/cache_service.py +0 -235
  105. /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/loading_spinner.html +0 -0
  106. /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/notification.html +0 -0
  107. /django_cfg/apps/payments/admin_interface/{templates → old}/payments/components/provider_card.html +0 -0
  108. /django_cfg/apps/payments/admin_interface/{templates → old}/payments/currency_converter.html +0 -0
  109. /django_cfg/apps/payments/admin_interface/{templates → old}/payments/payment_status.html +0 -0
  110. /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/css/dashboard.css +0 -0
  111. /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/js/components.js +0 -0
  112. /django_cfg/apps/payments/{static → admin_interface/old/static}/payments/js/utils.js +0 -0
  113. {django_cfg-1.3.1.dist-info → django_cfg-1.3.5.dist-info}/WHEEL +0 -0
  114. {django_cfg-1.3.1.dist-info → django_cfg-1.3.5.dist-info}/entry_points.txt +0 -0
  115. {django_cfg-1.3.1.dist-info → django_cfg-1.3.5.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,382 @@
1
+ {% extends 'payments/base.html' %}
2
+ {% load static %}
3
+ {% load payment_tags %}
4
+
5
+ {% block page_title %}{{ page_title }}{% endblock %}
6
+ {% block page_subtitle %}{{ page_subtitle }}{% endblock %}
7
+
8
+ {% block content %}
9
+ <div x-data="paymentList()" x-init="init()">
10
+ <!-- Filters and Search -->
11
+ <div class="payment-card mb-6">
12
+ <div class="payment-card-header">
13
+ <h3 class="font-medium text-gray-900 dark:text-white flex items-center">
14
+ <span class="material-icons-outlined mr-2">filter_list</span>
15
+ Filters & Search
16
+ </h3>
17
+ <button @click="resetFilters()"
18
+ class="text-sm text-blue-600 dark:text-blue-400 hover:text-blue-800 dark:hover:text-blue-300">
19
+ Reset
20
+ </button>
21
+ </div>
22
+ <div class="payment-card-content">
23
+ <div class="grid grid-cols-1 md:grid-cols-4 gap-4">
24
+ <!-- Search -->
25
+ <div>
26
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">Search</label>
27
+ <input type="text"
28
+ x-model="filters.search"
29
+ @input.debounce.300ms="applyFilters()"
30
+ placeholder="Payment ID, Amount..."
31
+ class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
32
+ </div>
33
+
34
+ <!-- Status Filter -->
35
+ <div>
36
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">Status</label>
37
+ <select x-model="filters.status" @change="applyFilters()"
38
+ class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
39
+ <option value="">All Statuses</option>
40
+ <option value="pending">Pending</option>
41
+ <option value="processing">Processing</option>
42
+ <option value="completed">Completed</option>
43
+ <option value="failed">Failed</option>
44
+ <option value="cancelled">Cancelled</option>
45
+ </select>
46
+ </div>
47
+
48
+ <!-- Provider Filter -->
49
+ <div>
50
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">Provider</label>
51
+ <select x-model="filters.provider" @change="applyFilters()"
52
+ class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
53
+ <option value="">All Providers</option>
54
+ <option value="nowpayments">NowPayments</option>
55
+ <option value="stripe">Stripe</option>
56
+ <option value="cryptapi">CryptAPI</option>
57
+ </select>
58
+ </div>
59
+
60
+ <!-- Date Range -->
61
+ <div>
62
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-1">Date Range</label>
63
+ <select x-model="filters.dateRange" @change="applyFilters()"
64
+ class="w-full px-3 py-2 border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
65
+ <option value="">All Time</option>
66
+ <option value="today">Today</option>
67
+ <option value="week">This Week</option>
68
+ <option value="month">This Month</option>
69
+ <option value="quarter">This Quarter</option>
70
+ </select>
71
+ </div>
72
+ </div>
73
+ </div>
74
+ </div>
75
+
76
+ <!-- Results Summary -->
77
+ <div class="flex items-center justify-between mb-6">
78
+ <div class="text-sm text-gray-600 dark:text-gray-400">
79
+ Showing <span x-text="payments.length"></span> payments
80
+ <span x-show="totalCount > payments.length" x-text="`of ${totalCount}`"></span>
81
+ </div>
82
+ <div class="flex items-center space-x-2">
83
+ <button @click="exportPayments()"
84
+ class="btn btn-outline text-sm">
85
+ <span class="material-icons-outlined mr-1 text-sm">download</span>
86
+ Export
87
+ </button>
88
+ <a href="{% url 'cfg_payments_admin:payment-create' %}"
89
+ class="btn btn-primary text-sm">
90
+ <span class="material-icons-outlined mr-1 text-sm">add</span>
91
+ New Payment
92
+ </a>
93
+ </div>
94
+ </div>
95
+
96
+ <!-- Payments Table -->
97
+ <div class="payment-card">
98
+ <div class="payment-card-content p-0">
99
+ <div class="overflow-x-auto">
100
+ <table class="w-full">
101
+ <thead class="bg-gray-50 dark:bg-gray-700">
102
+ <tr>
103
+ <th class="px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
104
+ Payment
105
+ </th>
106
+ <th class="px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
107
+ Amount
108
+ </th>
109
+ <th class="px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
110
+ Status
111
+ </th>
112
+ <th class="px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
113
+ Provider
114
+ </th>
115
+ <th class="px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
116
+ Date
117
+ </th>
118
+ <th class="px-6 py-3 text-right text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">
119
+ Actions
120
+ </th>
121
+ </tr>
122
+ </thead>
123
+ <tbody class="bg-white dark:bg-gray-800 divide-y divide-gray-200 dark:divide-gray-700">
124
+ <template x-for="payment in payments" :key="payment.id">
125
+ <tr class="hover:bg-gray-50 dark:hover:bg-gray-700">
126
+ <td class="px-6 py-4 whitespace-nowrap">
127
+ <div class="flex items-center">
128
+ <div class="w-8 h-8 rounded-full flex items-center justify-center mr-3"
129
+ :class="{
130
+ 'bg-green-100 dark:bg-green-900': payment.status === 'completed',
131
+ 'bg-yellow-100 dark:bg-yellow-900': payment.status === 'pending',
132
+ 'bg-blue-100 dark:bg-blue-900': payment.status === 'processing',
133
+ 'bg-red-100 dark:bg-red-900': ['failed', 'cancelled'].includes(payment.status)
134
+ }">
135
+ <span class="material-icons-outlined text-sm"
136
+ :class="{
137
+ 'text-green-600 dark:text-green-400': payment.status === 'completed',
138
+ 'text-yellow-600 dark:text-yellow-400': payment.status === 'pending',
139
+ 'text-blue-600 dark:text-blue-400': payment.status === 'processing',
140
+ 'text-red-600 dark:text-red-400': ['failed', 'cancelled'].includes(payment.status)
141
+ }"
142
+ x-text="getStatusIcon(payment.status)">
143
+ </span>
144
+ </div>
145
+ <div>
146
+ <div class="text-sm font-medium text-gray-900 dark:text-white" x-text="payment.external_id || payment.id.substring(0, 8)"></div>
147
+ <div class="text-sm text-gray-500 dark:text-gray-400" x-text="payment.currency_code"></div>
148
+ </div>
149
+ </div>
150
+ </td>
151
+ <td class="px-6 py-4 whitespace-nowrap">
152
+ <div class="text-sm font-medium text-gray-900 dark:text-white" x-text="`$${payment.amount_usd.toFixed(2)}`"></div>
153
+ <div class="text-sm text-gray-500 dark:text-gray-400" x-show="payment.amount_crypto" x-text="payment.amount_crypto + ' ' + payment.currency_code"></div>
154
+ </td>
155
+ <td class="px-6 py-4 whitespace-nowrap">
156
+ <span class="inline-flex px-2 py-1 text-xs font-semibold rounded-full"
157
+ :class="{
158
+ 'bg-green-100 text-green-800 dark:bg-green-900 dark:text-green-200': payment.status === 'completed',
159
+ 'bg-yellow-100 text-yellow-800 dark:bg-yellow-900 dark:text-yellow-200': payment.status === 'pending',
160
+ 'bg-blue-100 text-blue-800 dark:bg-blue-900 dark:text-blue-200': payment.status === 'processing',
161
+ 'bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-200': ['failed', 'cancelled'].includes(payment.status)
162
+ }"
163
+ x-text="payment.status.charAt(0).toUpperCase() + payment.status.slice(1)">
164
+ </span>
165
+ </td>
166
+ <td class="px-6 py-4 whitespace-nowrap text-sm text-gray-900 dark:text-white" x-text="payment.provider.charAt(0).toUpperCase() + payment.provider.slice(1)"></td>
167
+ <td class="px-6 py-4 whitespace-nowrap text-sm text-gray-500 dark:text-gray-400" x-text="formatDate(payment.created_at)"></td>
168
+ <td class="px-6 py-4 whitespace-nowrap text-right text-sm font-medium">
169
+ <div class="flex items-center justify-end space-x-2">
170
+ <a :href="`{% url 'cfg_payments_admin:payment-detail' '00000000-0000-0000-0000-000000000000' %}`.replace('00000000-0000-0000-0000-000000000000', payment.id)"
171
+ class="text-blue-600 dark:text-blue-400 hover:text-blue-800 dark:hover:text-blue-300">
172
+ <span class="material-icons-outlined text-sm">visibility</span>
173
+ </a>
174
+ <button @click="copyPaymentId(payment.id)"
175
+ class="text-gray-400 hover:text-gray-600 dark:hover:text-gray-200">
176
+ <span class="material-icons-outlined text-sm">content_copy</span>
177
+ </button>
178
+ </div>
179
+ </td>
180
+ </tr>
181
+ </template>
182
+ </tbody>
183
+ </table>
184
+
185
+ <!-- Empty State -->
186
+ <div x-show="payments.length === 0 && !loading" class="text-center py-12">
187
+ <span class="material-icons-outlined text-gray-400 text-6xl mb-4">payments</span>
188
+ <h3 class="text-lg font-medium text-gray-900 dark:text-white mb-2">No payments found</h3>
189
+ <p class="text-gray-500 dark:text-gray-400 mb-4">
190
+ <span x-show="hasActiveFilters()">Try adjusting your filters or</span>
191
+ <span x-show="!hasActiveFilters()">Get started by</span>
192
+ creating your first payment.
193
+ </p>
194
+ <a href="{% url 'cfg_payments_admin:payment-create' %}"
195
+ class="btn btn-primary">
196
+ <span class="material-icons-outlined mr-2">add</span>
197
+ Create Payment
198
+ </a>
199
+ </div>
200
+ </div>
201
+ </div>
202
+ </div>
203
+
204
+ <!-- Pagination -->
205
+ <div x-show="totalPages > 1" class="flex items-center justify-between mt-6">
206
+ <div class="text-sm text-gray-600 dark:text-gray-400">
207
+ Page <span x-text="currentPage"></span> of <span x-text="totalPages"></span>
208
+ </div>
209
+ <div class="flex items-center space-x-2">
210
+ <button @click="goToPage(currentPage - 1)"
211
+ :disabled="currentPage <= 1"
212
+ class="btn btn-outline text-sm"
213
+ :class="{ 'opacity-50 cursor-not-allowed': currentPage <= 1 }">
214
+ <span class="material-icons-outlined text-sm">chevron_left</span>
215
+ Previous
216
+ </button>
217
+ <button @click="goToPage(currentPage + 1)"
218
+ :disabled="currentPage >= totalPages"
219
+ class="btn btn-outline text-sm"
220
+ :class="{ 'opacity-50 cursor-not-allowed': currentPage >= totalPages }">
221
+ Next
222
+ <span class="material-icons-outlined text-sm">chevron_right</span>
223
+ </button>
224
+ </div>
225
+ </div>
226
+
227
+ <!-- Loading Spinner -->
228
+ <div x-show="loading" x-cloak class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
229
+ {% include 'payments/components/loading_spinner.html' %}
230
+ </div>
231
+
232
+ <!-- Notifications -->
233
+ <div x-show="notification.show" x-cloak
234
+ class="fixed top-4 right-4 z-50"
235
+ x-transition:enter="transition ease-out duration-300"
236
+ x-transition:enter-start="opacity-0 transform translate-x-full"
237
+ x-transition:enter-end="opacity-100 transform translate-x-0"
238
+ x-transition:leave="transition ease-in duration-200"
239
+ x-transition:leave-start="opacity-100 transform translate-x-0"
240
+ x-transition:leave-end="opacity-0 transform translate-x-full">
241
+ {% include 'payments/components/notification.html' with type="notification.type" message="notification.message" %}
242
+ </div>
243
+ </div>
244
+ {% endblock %}
245
+
246
+ {% block extra_js %}
247
+ <script>
248
+ function paymentList() {
249
+ return {
250
+ loading: false,
251
+ payments: [],
252
+ totalCount: 0,
253
+ currentPage: 1,
254
+ totalPages: 1,
255
+ filters: {
256
+ search: '',
257
+ status: '',
258
+ provider: '',
259
+ dateRange: ''
260
+ },
261
+ notification: {
262
+ show: false,
263
+ type: 'info',
264
+ message: ''
265
+ },
266
+
267
+ init() {
268
+ this.loadPayments();
269
+ },
270
+
271
+ async loadPayments() {
272
+ this.loading = true;
273
+ try {
274
+ // Simulate API call - replace with actual endpoint
275
+ await new Promise(resolve => setTimeout(resolve, 500));
276
+
277
+ // Mock data - replace with actual API response
278
+ this.payments = [
279
+ {
280
+ id: '123e4567-e89b-12d3-a456-426614174000',
281
+ external_id: 'PAY_001',
282
+ amount_usd: 100.00,
283
+ amount_crypto: '0.00234',
284
+ currency_code: 'BTC',
285
+ status: 'completed',
286
+ provider: 'nowpayments',
287
+ created_at: new Date().toISOString()
288
+ },
289
+ {
290
+ id: '123e4567-e89b-12d3-a456-426614174001',
291
+ external_id: 'PAY_002',
292
+ amount_usd: 50.00,
293
+ amount_crypto: null,
294
+ currency_code: 'USD',
295
+ status: 'pending',
296
+ provider: 'stripe',
297
+ created_at: new Date(Date.now() - 86400000).toISOString()
298
+ }
299
+ ];
300
+ this.totalCount = this.payments.length;
301
+ this.totalPages = Math.ceil(this.totalCount / 20);
302
+ } catch (error) {
303
+ this.showNotification('error', 'Failed to load payments');
304
+ } finally {
305
+ this.loading = false;
306
+ }
307
+ },
308
+
309
+ async applyFilters() {
310
+ this.currentPage = 1;
311
+ await this.loadPayments();
312
+ },
313
+
314
+ resetFilters() {
315
+ this.filters = {
316
+ search: '',
317
+ status: '',
318
+ provider: '',
319
+ dateRange: ''
320
+ };
321
+ this.applyFilters();
322
+ },
323
+
324
+ hasActiveFilters() {
325
+ return Object.values(this.filters).some(value => value !== '');
326
+ },
327
+
328
+ async goToPage(page) {
329
+ if (page >= 1 && page <= this.totalPages) {
330
+ this.currentPage = page;
331
+ await this.loadPayments();
332
+ }
333
+ },
334
+
335
+ getStatusIcon(status) {
336
+ const icons = {
337
+ completed: 'check_circle',
338
+ pending: 'schedule',
339
+ processing: 'sync',
340
+ failed: 'error',
341
+ cancelled: 'cancel'
342
+ };
343
+ return icons[status] || 'help';
344
+ },
345
+
346
+ formatDate(dateString) {
347
+ const date = new Date(dateString);
348
+ return date.toLocaleDateString() + ' ' + date.toLocaleTimeString([], { hour: '2-digit', minute: '2-digit' });
349
+ },
350
+
351
+ async copyPaymentId(paymentId) {
352
+ try {
353
+ await navigator.clipboard.writeText(paymentId);
354
+ this.showNotification('success', 'Payment ID copied to clipboard');
355
+ } catch (error) {
356
+ this.showNotification('error', 'Failed to copy payment ID');
357
+ }
358
+ },
359
+
360
+ async exportPayments() {
361
+ this.loading = true;
362
+ try {
363
+ // Simulate export - replace with actual implementation
364
+ await new Promise(resolve => setTimeout(resolve, 1000));
365
+ this.showNotification('success', 'Payments exported successfully');
366
+ } catch (error) {
367
+ this.showNotification('error', 'Failed to export payments');
368
+ } finally {
369
+ this.loading = false;
370
+ }
371
+ },
372
+
373
+ showNotification(type, message) {
374
+ this.notification = { show: true, type, message };
375
+ setTimeout(() => {
376
+ this.notification.show = false;
377
+ }, 5000);
378
+ }
379
+ };
380
+ }
381
+ </script>
382
+ {% endblock %}