django-cfg 1.2.31__py3-none-any.whl → 1.3.3__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 (264) hide show
  1. django_cfg/__init__.py +1 -1
  2. django_cfg/apps/api/health/views.py +4 -2
  3. django_cfg/apps/knowbase/config/settings.py +16 -15
  4. django_cfg/apps/payments/README.md +326 -0
  5. django_cfg/apps/payments/admin/__init__.py +20 -10
  6. django_cfg/apps/payments/admin/api_keys_admin.py +521 -237
  7. django_cfg/apps/payments/admin/balance_admin.py +592 -297
  8. django_cfg/apps/payments/admin/currencies_admin.py +526 -222
  9. django_cfg/apps/payments/admin/filters.py +306 -199
  10. django_cfg/apps/payments/admin/payments_admin.py +465 -70
  11. django_cfg/apps/payments/admin/subscriptions_admin.py +578 -128
  12. django_cfg/apps/payments/admin_interface/__init__.py +18 -0
  13. django_cfg/apps/payments/admin_interface/templates/payments/base.html +162 -0
  14. django_cfg/apps/payments/admin_interface/templates/payments/components/dev_tool_card.html +38 -0
  15. django_cfg/apps/payments/admin_interface/templates/payments/components/loading_spinner.html +16 -0
  16. django_cfg/apps/payments/admin_interface/templates/payments/components/notification.html +27 -0
  17. django_cfg/apps/payments/admin_interface/templates/payments/components/provider_card.html +86 -0
  18. django_cfg/apps/payments/admin_interface/templates/payments/components/status_card.html +39 -0
  19. django_cfg/apps/payments/admin_interface/templates/payments/currency_converter.html +382 -0
  20. django_cfg/apps/payments/admin_interface/templates/payments/payment_dashboard.html +300 -0
  21. django_cfg/apps/payments/admin_interface/templates/payments/payment_form.html +303 -0
  22. django_cfg/apps/payments/admin_interface/templates/payments/payment_list.html +382 -0
  23. django_cfg/apps/payments/admin_interface/templates/payments/payment_status.html +500 -0
  24. django_cfg/apps/payments/admin_interface/templates/payments/webhook_dashboard.html +594 -0
  25. django_cfg/apps/payments/admin_interface/views/__init__.py +23 -0
  26. django_cfg/apps/payments/admin_interface/views/payment_views.py +259 -0
  27. django_cfg/apps/payments/admin_interface/views/webhook_dashboard.py +37 -0
  28. django_cfg/apps/payments/apps.py +34 -9
  29. django_cfg/apps/payments/config/__init__.py +28 -51
  30. django_cfg/apps/payments/config/constance/__init__.py +22 -0
  31. django_cfg/apps/payments/config/constance/config_service.py +123 -0
  32. django_cfg/apps/payments/config/constance/fields.py +69 -0
  33. django_cfg/apps/payments/config/constance/settings.py +160 -0
  34. django_cfg/apps/payments/config/django_cfg_integration.py +202 -0
  35. django_cfg/apps/payments/config/helpers.py +130 -0
  36. django_cfg/apps/payments/management/__init__.py +1 -3
  37. django_cfg/apps/payments/management/commands/__init__.py +1 -3
  38. django_cfg/apps/payments/management/commands/cleanup_expired_data.py +419 -0
  39. django_cfg/apps/payments/management/commands/currency_stats.py +297 -225
  40. django_cfg/apps/payments/management/commands/manage_currencies.py +303 -151
  41. django_cfg/apps/payments/management/commands/manage_providers.py +333 -160
  42. django_cfg/apps/payments/management/commands/process_pending_payments.py +357 -0
  43. django_cfg/apps/payments/management/commands/test_providers.py +434 -0
  44. django_cfg/apps/payments/middleware/__init__.py +3 -1
  45. django_cfg/apps/payments/middleware/api_access.py +329 -222
  46. django_cfg/apps/payments/middleware/rate_limiting.py +342 -152
  47. django_cfg/apps/payments/middleware/usage_tracking.py +249 -240
  48. django_cfg/apps/payments/migrations/0001_initial.py +708 -536
  49. django_cfg/apps/payments/models/__init__.py +13 -18
  50. django_cfg/apps/payments/models/api_keys.py +121 -43
  51. django_cfg/apps/payments/models/balance.py +153 -115
  52. django_cfg/apps/payments/models/base.py +68 -15
  53. django_cfg/apps/payments/models/currencies.py +172 -148
  54. django_cfg/apps/payments/models/managers/__init__.py +44 -0
  55. django_cfg/apps/payments/models/managers/api_key_managers.py +329 -0
  56. django_cfg/apps/payments/models/managers/balance_managers.py +599 -0
  57. django_cfg/apps/payments/models/managers/currency_managers.py +385 -0
  58. django_cfg/apps/payments/models/managers/payment_managers.py +511 -0
  59. django_cfg/apps/payments/models/managers/subscription_managers.py +641 -0
  60. django_cfg/apps/payments/models/payments.py +235 -285
  61. django_cfg/apps/payments/models/subscriptions.py +257 -177
  62. django_cfg/apps/payments/models/tariffs.py +147 -40
  63. django_cfg/apps/payments/services/__init__.py +209 -56
  64. django_cfg/apps/payments/services/cache/__init__.py +6 -6
  65. django_cfg/apps/payments/services/cache_service/__init__.py +143 -0
  66. django_cfg/apps/payments/services/cache_service/api_key_cache.py +37 -0
  67. django_cfg/apps/payments/services/{cache/base.py → cache_service/interfaces.py} +3 -1
  68. django_cfg/apps/payments/services/cache_service/keys.py +49 -0
  69. django_cfg/apps/payments/services/cache_service/rate_limit_cache.py +47 -0
  70. django_cfg/apps/payments/services/cache_service/simple_cache.py +101 -0
  71. django_cfg/apps/payments/services/core/__init__.py +10 -6
  72. django_cfg/apps/payments/services/core/balance_service.py +435 -360
  73. django_cfg/apps/payments/services/core/base.py +166 -0
  74. django_cfg/apps/payments/services/core/currency_service.py +478 -0
  75. django_cfg/apps/payments/services/core/payment_service.py +371 -465
  76. django_cfg/apps/payments/services/core/subscription_service.py +425 -481
  77. django_cfg/apps/payments/services/core/webhook_service.py +410 -0
  78. django_cfg/apps/payments/services/integrations/__init__.py +29 -0
  79. django_cfg/apps/payments/services/integrations/ngrok_service.py +47 -0
  80. django_cfg/apps/payments/services/integrations/providers_config.py +107 -0
  81. django_cfg/apps/payments/services/providers/__init__.py +9 -14
  82. django_cfg/apps/payments/services/providers/base.py +234 -174
  83. django_cfg/apps/payments/services/providers/nowpayments.py +478 -0
  84. django_cfg/apps/payments/services/providers/registry.py +367 -301
  85. django_cfg/apps/payments/services/types/__init__.py +78 -0
  86. django_cfg/apps/payments/services/types/data.py +177 -0
  87. django_cfg/apps/payments/services/types/requests.py +150 -0
  88. django_cfg/apps/payments/services/types/responses.py +156 -0
  89. django_cfg/apps/payments/services/types/webhooks.py +232 -0
  90. django_cfg/apps/payments/signals/__init__.py +33 -8
  91. django_cfg/apps/payments/signals/api_key_signals.py +210 -129
  92. django_cfg/apps/payments/signals/balance_signals.py +174 -0
  93. django_cfg/apps/payments/signals/payment_signals.py +128 -103
  94. django_cfg/apps/payments/signals/subscription_signals.py +194 -142
  95. django_cfg/apps/payments/static/payments/css/components.css +380 -0
  96. django_cfg/apps/payments/static/payments/css/dashboard.css +188 -0
  97. django_cfg/apps/payments/static/payments/js/components.js +545 -0
  98. django_cfg/apps/payments/static/payments/js/utils.js +412 -0
  99. django_cfg/apps/payments/templatetags/__init__.py +1 -1
  100. django_cfg/apps/payments/templatetags/payment_tags.py +466 -0
  101. django_cfg/apps/payments/urls.py +45 -48
  102. django_cfg/apps/payments/urls_admin.py +33 -42
  103. django_cfg/apps/payments/views/api/__init__.py +101 -0
  104. django_cfg/apps/payments/views/api/api_keys.py +387 -0
  105. django_cfg/apps/payments/views/api/balances.py +381 -0
  106. django_cfg/apps/payments/views/api/base.py +298 -0
  107. django_cfg/apps/payments/views/api/currencies.py +402 -0
  108. django_cfg/apps/payments/views/api/payments.py +415 -0
  109. django_cfg/apps/payments/views/api/subscriptions.py +475 -0
  110. django_cfg/apps/payments/views/api/webhooks.py +476 -0
  111. django_cfg/apps/payments/views/serializers/__init__.py +99 -0
  112. django_cfg/apps/payments/views/serializers/api_keys.py +424 -0
  113. django_cfg/apps/payments/views/serializers/balances.py +300 -0
  114. django_cfg/apps/payments/views/serializers/currencies.py +335 -0
  115. django_cfg/apps/payments/views/serializers/payments.py +387 -0
  116. django_cfg/apps/payments/views/serializers/subscriptions.py +429 -0
  117. django_cfg/apps/payments/views/serializers/webhooks.py +137 -0
  118. django_cfg/config.py +1 -1
  119. django_cfg/core/config.py +40 -4
  120. django_cfg/core/generation.py +25 -4
  121. django_cfg/core/integration/README.md +363 -0
  122. django_cfg/core/integration/__init__.py +47 -0
  123. django_cfg/core/integration/commands_collector.py +239 -0
  124. django_cfg/core/integration/display/__init__.py +15 -0
  125. django_cfg/core/integration/display/base.py +157 -0
  126. django_cfg/core/integration/display/ngrok.py +164 -0
  127. django_cfg/core/integration/display/startup.py +815 -0
  128. django_cfg/core/integration/url_integration.py +123 -0
  129. django_cfg/core/integration/version_checker.py +160 -0
  130. django_cfg/management/commands/auto_generate.py +4 -0
  131. django_cfg/management/commands/check_settings.py +6 -0
  132. django_cfg/management/commands/clear_constance.py +5 -2
  133. django_cfg/management/commands/create_token.py +6 -0
  134. django_cfg/management/commands/list_urls.py +6 -0
  135. django_cfg/management/commands/migrate_all.py +6 -0
  136. django_cfg/management/commands/migrator.py +3 -0
  137. django_cfg/management/commands/rundramatiq.py +6 -0
  138. django_cfg/management/commands/runserver_ngrok.py +51 -29
  139. django_cfg/management/commands/script.py +6 -0
  140. django_cfg/management/commands/show_config.py +12 -2
  141. django_cfg/management/commands/show_urls.py +4 -0
  142. django_cfg/management/commands/superuser.py +6 -0
  143. django_cfg/management/commands/task_clear.py +4 -1
  144. django_cfg/management/commands/task_status.py +3 -1
  145. django_cfg/management/commands/test_email.py +3 -0
  146. django_cfg/management/commands/test_telegram.py +6 -0
  147. django_cfg/management/commands/test_twilio.py +6 -0
  148. django_cfg/management/commands/tree.py +6 -0
  149. django_cfg/management/commands/validate_config.py +155 -149
  150. django_cfg/models/constance.py +31 -11
  151. django_cfg/models/payments.py +175 -492
  152. django_cfg/modules/django_logger.py +160 -146
  153. django_cfg/modules/django_unfold/dashboard.py +64 -16
  154. django_cfg/registry/core.py +1 -0
  155. django_cfg/template_archive/django_sample.zip +0 -0
  156. django_cfg/utils/smart_defaults.py +227 -570
  157. django_cfg/utils/toolkit.py +51 -11
  158. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/METADATA +4 -1
  159. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/RECORD +162 -185
  160. django_cfg/apps/payments/__init__.py +0 -8
  161. django_cfg/apps/payments/admin/tariffs_admin.py +0 -199
  162. django_cfg/apps/payments/config/module.py +0 -70
  163. django_cfg/apps/payments/config/providers.py +0 -105
  164. django_cfg/apps/payments/config/settings.py +0 -96
  165. django_cfg/apps/payments/config/utils.py +0 -52
  166. django_cfg/apps/payments/decorators.py +0 -291
  167. django_cfg/apps/payments/management/commands/README.md +0 -146
  168. django_cfg/apps/payments/managers/__init__.py +0 -23
  169. django_cfg/apps/payments/managers/api_key_manager.py +0 -35
  170. django_cfg/apps/payments/managers/balance_manager.py +0 -361
  171. django_cfg/apps/payments/managers/currency_manager.py +0 -306
  172. django_cfg/apps/payments/managers/payment_manager.py +0 -192
  173. django_cfg/apps/payments/managers/subscription_manager.py +0 -37
  174. django_cfg/apps/payments/managers/tariff_manager.py +0 -29
  175. django_cfg/apps/payments/migrations/0002_network_providercurrency_and_more.py +0 -241
  176. django_cfg/apps/payments/migrations/0003_add_usd_rate_cache.py +0 -30
  177. django_cfg/apps/payments/models/events.py +0 -73
  178. django_cfg/apps/payments/serializers/__init__.py +0 -57
  179. django_cfg/apps/payments/serializers/api_keys.py +0 -51
  180. django_cfg/apps/payments/serializers/balance.py +0 -59
  181. django_cfg/apps/payments/serializers/currencies.py +0 -63
  182. django_cfg/apps/payments/serializers/payments.py +0 -62
  183. django_cfg/apps/payments/serializers/subscriptions.py +0 -71
  184. django_cfg/apps/payments/serializers/tariffs.py +0 -56
  185. django_cfg/apps/payments/services/billing/__init__.py +0 -8
  186. django_cfg/apps/payments/services/cache/simple_cache.py +0 -135
  187. django_cfg/apps/payments/services/core/fallback_service.py +0 -432
  188. django_cfg/apps/payments/services/internal_types.py +0 -461
  189. django_cfg/apps/payments/services/middleware/__init__.py +0 -8
  190. django_cfg/apps/payments/services/monitoring/__init__.py +0 -22
  191. django_cfg/apps/payments/services/monitoring/api_schemas.py +0 -76
  192. django_cfg/apps/payments/services/monitoring/provider_health.py +0 -372
  193. django_cfg/apps/payments/services/providers/cryptapi/__init__.py +0 -4
  194. django_cfg/apps/payments/services/providers/cryptapi/config.py +0 -8
  195. django_cfg/apps/payments/services/providers/cryptapi/models.py +0 -192
  196. django_cfg/apps/payments/services/providers/cryptapi/provider.py +0 -439
  197. django_cfg/apps/payments/services/providers/cryptomus/__init__.py +0 -4
  198. django_cfg/apps/payments/services/providers/cryptomus/models.py +0 -176
  199. django_cfg/apps/payments/services/providers/cryptomus/provider.py +0 -429
  200. django_cfg/apps/payments/services/providers/cryptomus/provider_v2.py +0 -564
  201. django_cfg/apps/payments/services/providers/models/__init__.py +0 -34
  202. django_cfg/apps/payments/services/providers/models/currencies.py +0 -190
  203. django_cfg/apps/payments/services/providers/nowpayments/__init__.py +0 -4
  204. django_cfg/apps/payments/services/providers/nowpayments/models.py +0 -196
  205. django_cfg/apps/payments/services/providers/nowpayments/provider.py +0 -380
  206. django_cfg/apps/payments/services/providers/stripe/__init__.py +0 -4
  207. django_cfg/apps/payments/services/providers/stripe/models.py +0 -184
  208. django_cfg/apps/payments/services/providers/stripe/provider.py +0 -109
  209. django_cfg/apps/payments/services/security/__init__.py +0 -34
  210. django_cfg/apps/payments/services/security/error_handler.py +0 -635
  211. django_cfg/apps/payments/services/security/payment_notifications.py +0 -342
  212. django_cfg/apps/payments/services/security/webhook_validator.py +0 -474
  213. django_cfg/apps/payments/static/payments/css/payments.css +0 -340
  214. django_cfg/apps/payments/static/payments/js/notifications.js +0 -202
  215. django_cfg/apps/payments/static/payments/js/payment-utils.js +0 -318
  216. django_cfg/apps/payments/static/payments/js/theme.js +0 -86
  217. django_cfg/apps/payments/tasks/__init__.py +0 -12
  218. django_cfg/apps/payments/tasks/webhook_processing.py +0 -177
  219. django_cfg/apps/payments/templates/admin/payments/currency/change_list.html +0 -50
  220. django_cfg/apps/payments/templates/payments/base.html +0 -182
  221. django_cfg/apps/payments/templates/payments/components/payment_card.html +0 -201
  222. django_cfg/apps/payments/templates/payments/components/payment_qr_code.html +0 -109
  223. django_cfg/apps/payments/templates/payments/components/progress_bar.html +0 -43
  224. django_cfg/apps/payments/templates/payments/components/provider_stats.html +0 -40
  225. django_cfg/apps/payments/templates/payments/components/status_badge.html +0 -34
  226. django_cfg/apps/payments/templates/payments/components/status_overview.html +0 -148
  227. django_cfg/apps/payments/templates/payments/dashboard.html +0 -258
  228. django_cfg/apps/payments/templates/payments/dashboard_simple_test.html +0 -35
  229. django_cfg/apps/payments/templates/payments/payment_create.html +0 -579
  230. django_cfg/apps/payments/templates/payments/payment_detail.html +0 -373
  231. django_cfg/apps/payments/templates/payments/payment_list.html +0 -354
  232. django_cfg/apps/payments/templates/payments/stats.html +0 -261
  233. django_cfg/apps/payments/templates/payments/test.html +0 -213
  234. django_cfg/apps/payments/templatetags/payments_tags.py +0 -315
  235. django_cfg/apps/payments/utils/__init__.py +0 -43
  236. django_cfg/apps/payments/utils/billing_utils.py +0 -342
  237. django_cfg/apps/payments/utils/config_utils.py +0 -239
  238. django_cfg/apps/payments/utils/middleware_utils.py +0 -228
  239. django_cfg/apps/payments/utils/validation_utils.py +0 -94
  240. django_cfg/apps/payments/views/__init__.py +0 -63
  241. django_cfg/apps/payments/views/api_key_views.py +0 -164
  242. django_cfg/apps/payments/views/balance_views.py +0 -75
  243. django_cfg/apps/payments/views/currency_views.py +0 -122
  244. django_cfg/apps/payments/views/payment_views.py +0 -149
  245. django_cfg/apps/payments/views/subscription_views.py +0 -135
  246. django_cfg/apps/payments/views/tariff_views.py +0 -131
  247. django_cfg/apps/payments/views/templates/__init__.py +0 -25
  248. django_cfg/apps/payments/views/templates/ajax.py +0 -451
  249. django_cfg/apps/payments/views/templates/base.py +0 -212
  250. django_cfg/apps/payments/views/templates/dashboard.py +0 -60
  251. django_cfg/apps/payments/views/templates/payment_detail.py +0 -102
  252. django_cfg/apps/payments/views/templates/payment_management.py +0 -158
  253. django_cfg/apps/payments/views/templates/qr_code.py +0 -174
  254. django_cfg/apps/payments/views/templates/stats.py +0 -244
  255. django_cfg/apps/payments/views/templates/utils.py +0 -181
  256. django_cfg/apps/payments/views/webhook_views.py +0 -266
  257. django_cfg/apps/payments/viewsets.py +0 -66
  258. django_cfg/core/integration.py +0 -160
  259. django_cfg/template_archive/.gitignore +0 -1
  260. django_cfg/template_archive/__init__.py +0 -0
  261. django_cfg/urls.py +0 -33
  262. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/WHEEL +0 -0
  263. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.dist-info}/entry_points.txt +0 -0
  264. {django_cfg-1.2.31.dist-info → django_cfg-1.3.3.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="currencyConverter()" x-init="init()">
10
+ <!-- Converter Tool -->
11
+ <div class="max-w-4xl mx-auto">
12
+ <div class="payment-card mb-6">
13
+ <div class="payment-card-header">
14
+ <h3 class="font-medium text-gray-900 dark:text-white flex items-center">
15
+ <span class="material-icons-outlined mr-2">currency_exchange</span>
16
+ Currency Converter
17
+ </h3>
18
+ <button @click="swapCurrencies()"
19
+ class="text-blue-600 dark:text-blue-400 hover:text-blue-800 dark:hover:text-blue-300">
20
+ <span class="material-icons-outlined">swap_horiz</span>
21
+ </button>
22
+ </div>
23
+ <div class="payment-card-content">
24
+ <div class="grid grid-cols-1 lg:grid-cols-2 gap-6">
25
+ <!-- From Currency -->
26
+ <div>
27
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">From</label>
28
+ <div class="space-y-3">
29
+ <select x-model="fromCurrency" @change="convert()"
30
+ 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">
31
+ <template x-for="currency in currencies" :key="currency.code">
32
+ <option :value="currency.code" x-text="`${currency.code} - ${currency.name}`"></option>
33
+ </template>
34
+ </select>
35
+ <div class="relative">
36
+ <input type="number"
37
+ x-model="fromAmount"
38
+ @input.debounce.500ms="convert()"
39
+ step="0.00000001"
40
+ placeholder="Enter amount"
41
+ class="w-full px-3 py-3 text-lg border border-gray-300 dark:border-gray-600 rounded-md bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
42
+ <div class="absolute inset-y-0 right-0 pr-3 flex items-center">
43
+ <span class="text-gray-500 dark:text-gray-400 text-sm" x-text="fromCurrency"></span>
44
+ </div>
45
+ </div>
46
+ </div>
47
+ </div>
48
+
49
+ <!-- To Currency -->
50
+ <div>
51
+ <label class="block text-sm font-medium text-gray-700 dark:text-gray-300 mb-2">To</label>
52
+ <div class="space-y-3">
53
+ <select x-model="toCurrency" @change="convert()"
54
+ 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">
55
+ <template x-for="currency in currencies" :key="currency.code">
56
+ <option :value="currency.code" x-text="`${currency.code} - ${currency.name}`"></option>
57
+ </template>
58
+ </select>
59
+ <div class="relative">
60
+ <input type="text"
61
+ :value="toAmount"
62
+ readonly
63
+ placeholder="Converted amount"
64
+ class="w-full px-3 py-3 text-lg border border-gray-300 dark:border-gray-600 rounded-md bg-gray-50 dark:bg-gray-600 text-gray-900 dark:text-white">
65
+ <div class="absolute inset-y-0 right-0 pr-3 flex items-center">
66
+ <span class="text-gray-500 dark:text-gray-400 text-sm" x-text="toCurrency"></span>
67
+ </div>
68
+ </div>
69
+ </div>
70
+ </div>
71
+ </div>
72
+
73
+ <!-- Exchange Rate Info -->
74
+ <div x-show="exchangeRate" class="mt-6 p-4 bg-blue-50 dark:bg-blue-900/20 rounded-lg">
75
+ <div class="flex items-center justify-between">
76
+ <div class="flex items-center space-x-2">
77
+ <span class="material-icons-outlined text-blue-600 dark:text-blue-400">trending_up</span>
78
+ <span class="text-sm font-medium text-gray-900 dark:text-white">Exchange Rate</span>
79
+ </div>
80
+ <div class="text-sm text-gray-600 dark:text-gray-400" x-text="lastUpdated"></div>
81
+ </div>
82
+ <div class="mt-2">
83
+ <span class="text-lg font-semibold text-gray-900 dark:text-white">
84
+ 1 <span x-text="fromCurrency"></span> = <span x-text="exchangeRate"></span> <span x-text="toCurrency"></span>
85
+ </span>
86
+ </div>
87
+ </div>
88
+
89
+ <!-- Quick Actions -->
90
+ <div class="mt-6 flex flex-wrap gap-2">
91
+ <template x-for="amount in quickAmounts" :key="amount">
92
+ <button @click="setQuickAmount(amount)"
93
+ class="px-3 py-1 text-sm bg-gray-100 dark:bg-gray-700 text-gray-700 dark:text-gray-300 rounded-md hover:bg-gray-200 dark:hover:bg-gray-600">
94
+ <span x-text="amount"></span> <span x-text="fromCurrency"></span>
95
+ </button>
96
+ </template>
97
+ </div>
98
+ </div>
99
+ </div>
100
+
101
+ <!-- Popular Conversions -->
102
+ <div class="grid grid-cols-1 lg:grid-cols-2 gap-6 mb-6">
103
+ <!-- Popular Pairs -->
104
+ <div class="payment-card">
105
+ <div class="payment-card-header">
106
+ <h3 class="font-medium text-gray-900 dark:text-white flex items-center">
107
+ <span class="material-icons-outlined mr-2">star</span>
108
+ Popular Pairs
109
+ </h3>
110
+ </div>
111
+ <div class="payment-card-content">
112
+ <div class="space-y-3">
113
+ <template x-for="pair in popularPairs" :key="pair.from + pair.to">
114
+ <div class="flex items-center justify-between p-3 bg-gray-50 dark:bg-gray-700 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-600 cursor-pointer"
115
+ @click="selectPair(pair.from, pair.to)">
116
+ <div class="flex items-center space-x-3">
117
+ <div class="flex items-center space-x-1">
118
+ <span class="text-sm font-medium text-gray-900 dark:text-white" x-text="pair.from"></span>
119
+ <span class="material-icons-outlined text-gray-400 text-sm">arrow_forward</span>
120
+ <span class="text-sm font-medium text-gray-900 dark:text-white" x-text="pair.to"></span>
121
+ </div>
122
+ </div>
123
+ <div class="text-sm text-gray-600 dark:text-gray-400" x-text="pair.rate"></div>
124
+ </div>
125
+ </template>
126
+ </div>
127
+ </div>
128
+ </div>
129
+
130
+ <!-- Rate History Chart -->
131
+ <div class="payment-card">
132
+ <div class="payment-card-header">
133
+ <h3 class="font-medium text-gray-900 dark:text-white flex items-center">
134
+ <span class="material-icons-outlined mr-2">show_chart</span>
135
+ Rate History (24h)
136
+ </h3>
137
+ <select x-model="chartPeriod" @change="updateChart()"
138
+ class="text-sm border border-gray-300 dark:border-gray-600 rounded px-2 py-1 bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
139
+ <option value="1h">1 Hour</option>
140
+ <option value="24h">24 Hours</option>
141
+ <option value="7d">7 Days</option>
142
+ <option value="30d">30 Days</option>
143
+ </select>
144
+ </div>
145
+ <div class="payment-card-content">
146
+ <div class="h-48 flex items-center justify-center bg-gray-50 dark:bg-gray-700 rounded-lg">
147
+ <div class="text-center">
148
+ <span class="material-icons-outlined text-gray-400 text-4xl mb-2">show_chart</span>
149
+ <p class="text-gray-500 dark:text-gray-400 text-sm">Rate history chart</p>
150
+ <p class="text-gray-400 dark:text-gray-500 text-xs">Chart visualization would be implemented here</p>
151
+ </div>
152
+ </div>
153
+ </div>
154
+ </div>
155
+ </div>
156
+
157
+ <!-- Supported Currencies -->
158
+ <div class="payment-card">
159
+ <div class="payment-card-header">
160
+ <h3 class="font-medium text-gray-900 dark:text-white flex items-center">
161
+ <span class="material-icons-outlined mr-2">account_balance</span>
162
+ Supported Currencies
163
+ </h3>
164
+ <div class="flex items-center space-x-2">
165
+ <input type="text"
166
+ x-model="currencySearch"
167
+ placeholder="Search currencies..."
168
+ class="text-sm px-3 py-1 border border-gray-300 dark:border-gray-600 rounded bg-white dark:bg-gray-700 text-gray-900 dark:text-white">
169
+ <span class="text-sm text-gray-500 dark:text-gray-400" x-text="`${filteredCurrencies.length} currencies`"></span>
170
+ </div>
171
+ </div>
172
+ <div class="payment-card-content">
173
+ <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
174
+ <template x-for="currency in filteredCurrencies.slice(0, 12)" :key="currency.code">
175
+ <div class="flex items-center justify-between p-3 bg-gray-50 dark:bg-gray-700 rounded-lg hover:bg-gray-100 dark:hover:bg-gray-600 cursor-pointer"
176
+ @click="selectCurrency(currency.code)">
177
+ <div class="flex items-center space-x-3">
178
+ <div class="w-8 h-8 bg-blue-100 dark:bg-blue-900 rounded-full flex items-center justify-center">
179
+ <span class="text-xs font-bold text-blue-600 dark:text-blue-400" x-text="currency.code.substring(0, 2)"></span>
180
+ </div>
181
+ <div>
182
+ <div class="text-sm font-medium text-gray-900 dark:text-white" x-text="currency.code"></div>
183
+ <div class="text-xs text-gray-500 dark:text-gray-400" x-text="currency.name"></div>
184
+ </div>
185
+ </div>
186
+ <div class="text-xs text-gray-400" x-text="currency.type"></div>
187
+ </div>
188
+ </template>
189
+ </div>
190
+ <div x-show="filteredCurrencies.length > 12" class="mt-4 text-center">
191
+ <button @click="showAllCurrencies = !showAllCurrencies"
192
+ class="text-sm text-blue-600 dark:text-blue-400 hover:text-blue-800 dark:hover:text-blue-300">
193
+ <span x-show="!showAllCurrencies">Show all currencies</span>
194
+ <span x-show="showAllCurrencies">Show less</span>
195
+ </button>
196
+ </div>
197
+ </div>
198
+ </div>
199
+ </div>
200
+
201
+ <!-- Loading Spinner -->
202
+ <div x-show="loading" x-cloak class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center z-50">
203
+ {% include 'payments/components/loading_spinner.html' %}
204
+ </div>
205
+
206
+ <!-- Notifications -->
207
+ <div x-show="notification.show" x-cloak
208
+ class="fixed top-4 right-4 z-50"
209
+ x-transition:enter="transition ease-out duration-300"
210
+ x-transition:enter-start="opacity-0 transform translate-x-full"
211
+ x-transition:enter-end="opacity-100 transform translate-x-0"
212
+ x-transition:leave="transition ease-in duration-200"
213
+ x-transition:leave-start="opacity-100 transform translate-x-0"
214
+ x-transition:leave-end="opacity-0 transform translate-x-full">
215
+ {% include 'payments/components/notification.html' with type="notification.type" message="notification.message" %}
216
+ </div>
217
+ </div>
218
+ {% endblock %}
219
+
220
+ {% block extra_js %}
221
+ <script>
222
+ function currencyConverter() {
223
+ return {
224
+ loading: false,
225
+ fromCurrency: 'USD',
226
+ toCurrency: 'BTC',
227
+ fromAmount: 100,
228
+ toAmount: '',
229
+ exchangeRate: '',
230
+ lastUpdated: '',
231
+ currencySearch: '',
232
+ showAllCurrencies: false,
233
+ chartPeriod: '24h',
234
+ currencies: [
235
+ { code: 'USD', name: 'US Dollar', type: 'Fiat' },
236
+ { code: 'EUR', name: 'Euro', type: 'Fiat' },
237
+ { code: 'GBP', name: 'British Pound', type: 'Fiat' },
238
+ { code: 'BTC', name: 'Bitcoin', type: 'Crypto' },
239
+ { code: 'ETH', name: 'Ethereum', type: 'Crypto' },
240
+ { code: 'LTC', name: 'Litecoin', type: 'Crypto' },
241
+ { code: 'XRP', name: 'Ripple', type: 'Crypto' },
242
+ { code: 'ADA', name: 'Cardano', type: 'Crypto' },
243
+ { code: 'DOT', name: 'Polkadot', type: 'Crypto' },
244
+ { code: 'LINK', name: 'Chainlink', type: 'Crypto' }
245
+ ],
246
+ popularPairs: [
247
+ { from: 'USD', to: 'BTC', rate: '0.000023' },
248
+ { from: 'USD', to: 'ETH', rate: '0.00041' },
249
+ { from: 'EUR', to: 'USD', rate: '1.08' },
250
+ { from: 'BTC', to: 'USD', rate: '43,250' },
251
+ { from: 'ETH', to: 'USD', rate: '2,450' }
252
+ ],
253
+ quickAmounts: [1, 10, 100, 1000, 10000],
254
+ notification: {
255
+ show: false,
256
+ type: 'info',
257
+ message: ''
258
+ },
259
+
260
+ init() {
261
+ this.convert();
262
+ },
263
+
264
+ get filteredCurrencies() {
265
+ if (!this.currencySearch) return this.currencies;
266
+ const search = this.currencySearch.toLowerCase();
267
+ return this.currencies.filter(currency =>
268
+ currency.code.toLowerCase().includes(search) ||
269
+ currency.name.toLowerCase().includes(search)
270
+ );
271
+ },
272
+
273
+ async convert() {
274
+ if (!this.fromAmount || this.fromAmount <= 0) {
275
+ this.toAmount = '';
276
+ this.exchangeRate = '';
277
+ return;
278
+ }
279
+
280
+ this.loading = true;
281
+ try {
282
+ // Simulate API call - replace with actual conversion endpoint
283
+ await new Promise(resolve => setTimeout(resolve, 300));
284
+
285
+ // Mock conversion rates
286
+ const rates = {
287
+ 'USD_BTC': 0.000023,
288
+ 'USD_ETH': 0.00041,
289
+ 'BTC_USD': 43250,
290
+ 'ETH_USD': 2450,
291
+ 'EUR_USD': 1.08,
292
+ 'USD_EUR': 0.93
293
+ };
294
+
295
+ const rateKey = `${this.fromCurrency}_${this.toCurrency}`;
296
+ const reverseRateKey = `${this.toCurrency}_${this.fromCurrency}`;
297
+
298
+ let rate = rates[rateKey];
299
+ if (!rate && rates[reverseRateKey]) {
300
+ rate = 1 / rates[reverseRateKey];
301
+ }
302
+ if (!rate) rate = 1; // Fallback
303
+
304
+ const convertedAmount = this.fromAmount * rate;
305
+ this.toAmount = this.formatAmount(convertedAmount);
306
+ this.exchangeRate = this.formatRate(rate);
307
+ this.lastUpdated = 'Updated ' + new Date().toLocaleTimeString();
308
+
309
+ } catch (error) {
310
+ this.showNotification('error', 'Failed to convert currency');
311
+ } finally {
312
+ this.loading = false;
313
+ }
314
+ },
315
+
316
+ swapCurrencies() {
317
+ const temp = this.fromCurrency;
318
+ this.fromCurrency = this.toCurrency;
319
+ this.toCurrency = temp;
320
+
321
+ if (this.toAmount && this.toAmount !== '') {
322
+ this.fromAmount = parseFloat(this.toAmount);
323
+ }
324
+
325
+ this.convert();
326
+ },
327
+
328
+ selectPair(from, to) {
329
+ this.fromCurrency = from;
330
+ this.toCurrency = to;
331
+ this.convert();
332
+ },
333
+
334
+ selectCurrency(code) {
335
+ if (this.fromCurrency === code) {
336
+ this.toCurrency = code;
337
+ } else {
338
+ this.fromCurrency = code;
339
+ }
340
+ this.convert();
341
+ },
342
+
343
+ setQuickAmount(amount) {
344
+ this.fromAmount = amount;
345
+ this.convert();
346
+ },
347
+
348
+ updateChart() {
349
+ // Chart update logic would go here
350
+ this.showNotification('info', `Chart updated for ${this.chartPeriod} period`);
351
+ },
352
+
353
+ formatAmount(amount) {
354
+ if (amount >= 1) {
355
+ return amount.toFixed(2);
356
+ } else if (amount >= 0.01) {
357
+ return amount.toFixed(4);
358
+ } else {
359
+ return amount.toFixed(8);
360
+ }
361
+ },
362
+
363
+ formatRate(rate) {
364
+ if (rate >= 1) {
365
+ return rate.toFixed(2);
366
+ } else if (rate >= 0.01) {
367
+ return rate.toFixed(4);
368
+ } else {
369
+ return rate.toFixed(8);
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 %}