sqa 0.0.24 → 0.0.32

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 (203) hide show
  1. checksums.yaml +4 -4
  2. data/.goose/memory/development.txt +3 -0
  3. data/.semver +6 -0
  4. data/ARCHITECTURE.md +648 -0
  5. data/CHANGELOG.md +95 -0
  6. data/CLAUDE.md +674 -0
  7. data/COMMITS.md +196 -0
  8. data/DATAFRAME_ARCHITECTURE_REVIEW.md +421 -0
  9. data/NEXT-STEPS.md +154 -0
  10. data/README.md +839 -265
  11. data/TASKS.md +358 -0
  12. data/TEST_RESULTS.md +140 -0
  13. data/TODO.md +42 -0
  14. data/_notes.txt +25 -0
  15. data/bin/sqa-console +11 -0
  16. data/data/talk_talk.json +103284 -0
  17. data/develop_summary.md +313 -0
  18. data/docs/advanced/backtesting.md +206 -0
  19. data/docs/advanced/ensemble.md +68 -0
  20. data/docs/advanced/fpop.md +153 -0
  21. data/docs/advanced/index.md +112 -0
  22. data/docs/advanced/multi-timeframe.md +67 -0
  23. data/docs/advanced/pattern-matcher.md +75 -0
  24. data/docs/advanced/portfolio-optimizer.md +79 -0
  25. data/docs/advanced/portfolio.md +166 -0
  26. data/docs/advanced/risk-management.md +210 -0
  27. data/docs/advanced/strategy-generator.md +158 -0
  28. data/docs/advanced/streaming.md +209 -0
  29. data/docs/ai_and_ml.md +80 -0
  30. data/docs/api/dataframe.md +1114 -0
  31. data/docs/api/index.md +126 -0
  32. data/docs/assets/css/custom.css +88 -0
  33. data/docs/assets/images/sqa.jpg +0 -0
  34. data/docs/assets/js/mathjax.js +18 -0
  35. data/docs/concepts/index.md +60 -0
  36. data/docs/contributing/index.md +60 -0
  37. data/docs/data-sources/index.md +66 -0
  38. data/docs/data_frame.md +316 -97
  39. data/docs/factors_that_impact_price.md +26 -0
  40. data/docs/finviz.md +11 -0
  41. data/docs/fx_pro_bit.md +25 -0
  42. data/docs/genetic_programming.md +104 -0
  43. data/docs/getting-started/index.md +107 -0
  44. data/docs/getting-started/installation.md +229 -0
  45. data/docs/getting-started/quick-start.md +244 -0
  46. data/docs/i_gotta_an_idea.md +22 -0
  47. data/docs/index.md +161 -0
  48. data/docs/indicators/index.md +97 -0
  49. data/docs/indicators.md +110 -24
  50. data/docs/options.md +8 -0
  51. data/docs/strategies/bollinger-bands.md +146 -0
  52. data/docs/strategies/consensus.md +64 -0
  53. data/docs/strategies/custom.md +310 -0
  54. data/docs/strategies/ema.md +53 -0
  55. data/docs/strategies/index.md +92 -0
  56. data/docs/strategies/kbs.md +164 -0
  57. data/docs/strategies/macd.md +96 -0
  58. data/docs/strategies/market-profile.md +54 -0
  59. data/docs/strategies/mean-reversion.md +58 -0
  60. data/docs/strategies/rsi.md +95 -0
  61. data/docs/strategies/sma.md +55 -0
  62. data/docs/strategies/stochastic.md +63 -0
  63. data/docs/strategies/volume-breakout.md +54 -0
  64. data/docs/tags.md +7 -0
  65. data/examples/README.md +354 -0
  66. data/examples/advanced_features_example.rb +350 -0
  67. data/examples/fpop_analysis_example.rb +191 -0
  68. data/examples/genetic_programming_example.rb +148 -0
  69. data/examples/kbs_strategy_example.rb +208 -0
  70. data/examples/pattern_context_example.rb +300 -0
  71. data/examples/rails_app/Gemfile +34 -0
  72. data/examples/rails_app/README.md +416 -0
  73. data/examples/rails_app/app/assets/javascripts/application.js +107 -0
  74. data/examples/rails_app/app/assets/stylesheets/application.css +659 -0
  75. data/examples/rails_app/app/controllers/analysis_controller.rb +11 -0
  76. data/examples/rails_app/app/controllers/api/v1/stocks_controller.rb +227 -0
  77. data/examples/rails_app/app/controllers/application_controller.rb +22 -0
  78. data/examples/rails_app/app/controllers/backtest_controller.rb +11 -0
  79. data/examples/rails_app/app/controllers/dashboard_controller.rb +21 -0
  80. data/examples/rails_app/app/controllers/portfolio_controller.rb +7 -0
  81. data/examples/rails_app/app/views/analysis/show.html.erb +209 -0
  82. data/examples/rails_app/app/views/backtest/show.html.erb +171 -0
  83. data/examples/rails_app/app/views/dashboard/index.html.erb +118 -0
  84. data/examples/rails_app/app/views/dashboard/show.html.erb +408 -0
  85. data/examples/rails_app/app/views/errors/show.html.erb +17 -0
  86. data/examples/rails_app/app/views/layouts/application.html.erb +60 -0
  87. data/examples/rails_app/app/views/portfolio/index.html.erb +33 -0
  88. data/examples/rails_app/bin/rails +6 -0
  89. data/examples/rails_app/config/application.rb +45 -0
  90. data/examples/rails_app/config/boot.rb +5 -0
  91. data/examples/rails_app/config/database.yml +18 -0
  92. data/examples/rails_app/config/environment.rb +11 -0
  93. data/examples/rails_app/config/routes.rb +26 -0
  94. data/examples/rails_app/config.ru +8 -0
  95. data/examples/realtime_stream_example.rb +274 -0
  96. data/examples/sinatra_app/Gemfile +42 -0
  97. data/examples/sinatra_app/Gemfile.lock +268 -0
  98. data/examples/sinatra_app/QUICKSTART.md +169 -0
  99. data/examples/sinatra_app/README.md +471 -0
  100. data/examples/sinatra_app/RUNNING_WITHOUT_TALIB.md +90 -0
  101. data/examples/sinatra_app/TROUBLESHOOTING.md +95 -0
  102. data/examples/sinatra_app/app.rb +404 -0
  103. data/examples/sinatra_app/config.ru +5 -0
  104. data/examples/sinatra_app/public/css/style.css +723 -0
  105. data/examples/sinatra_app/public/debug_macd.html +82 -0
  106. data/examples/sinatra_app/public/js/app.js +107 -0
  107. data/examples/sinatra_app/start.sh +53 -0
  108. data/examples/sinatra_app/views/analyze.erb +306 -0
  109. data/examples/sinatra_app/views/backtest.erb +325 -0
  110. data/examples/sinatra_app/views/dashboard.erb +831 -0
  111. data/examples/sinatra_app/views/error.erb +58 -0
  112. data/examples/sinatra_app/views/index.erb +118 -0
  113. data/examples/sinatra_app/views/layout.erb +61 -0
  114. data/examples/sinatra_app/views/portfolio.erb +43 -0
  115. data/examples/strategy_generator_example.rb +346 -0
  116. data/hsa_portfolio.csv +11 -0
  117. data/justfile +0 -0
  118. data/lib/api/alpha_vantage_api.rb +462 -0
  119. data/lib/sqa/backtest.rb +329 -0
  120. data/lib/sqa/data_frame/alpha_vantage.rb +51 -63
  121. data/lib/sqa/data_frame/data.rb +92 -0
  122. data/lib/sqa/data_frame/yahoo_finance.rb +35 -43
  123. data/lib/sqa/data_frame.rb +154 -243
  124. data/lib/sqa/ensemble.rb +359 -0
  125. data/lib/sqa/fpop.rb +199 -0
  126. data/lib/sqa/gp.rb +259 -0
  127. data/lib/sqa/indicator.rb +16 -6
  128. data/lib/sqa/init.rb +15 -8
  129. data/lib/sqa/market_regime.rb +240 -0
  130. data/lib/sqa/multi_timeframe.rb +379 -0
  131. data/lib/sqa/pattern_matcher.rb +497 -0
  132. data/lib/sqa/portfolio.rb +260 -6
  133. data/lib/sqa/portfolio_optimizer.rb +377 -0
  134. data/lib/sqa/risk_manager.rb +442 -0
  135. data/lib/sqa/seasonal_analyzer.rb +209 -0
  136. data/lib/sqa/sector_analyzer.rb +300 -0
  137. data/lib/sqa/stock.rb +131 -127
  138. data/lib/sqa/strategy/bollinger_bands.rb +42 -0
  139. data/lib/sqa/strategy/consensus.rb +5 -2
  140. data/lib/sqa/strategy/kbs_strategy.rb +470 -0
  141. data/lib/sqa/strategy/macd.rb +46 -0
  142. data/lib/sqa/strategy/mp.rb +1 -1
  143. data/lib/sqa/strategy/stochastic.rb +60 -0
  144. data/lib/sqa/strategy/volume_breakout.rb +57 -0
  145. data/lib/sqa/strategy.rb +5 -0
  146. data/lib/sqa/strategy_generator.rb +947 -0
  147. data/lib/sqa/stream.rb +361 -0
  148. data/lib/sqa/ticker.rb +9 -2
  149. data/lib/sqa/version.rb +1 -7
  150. data/lib/sqa.rb +35 -20
  151. data/main.just +81 -0
  152. data/mkdocs.yml +252 -0
  153. data/trace.log +0 -0
  154. metadata +265 -69
  155. data/bin/sqa +0 -6
  156. data/docs/alpha_vantage_technical_indicators.md +0 -62
  157. data/docs/average_true_range.md +0 -9
  158. data/docs/bollinger_bands.md +0 -15
  159. data/docs/candlestick_pattern_recognizer.md +0 -4
  160. data/docs/donchian_channel.md +0 -5
  161. data/docs/double_top_bottom_pattern.md +0 -3
  162. data/docs/exponential_moving_average.md +0 -19
  163. data/docs/fibonacci_retracement.md +0 -30
  164. data/docs/head_and_shoulders_pattern.md +0 -3
  165. data/docs/market_profile.md +0 -4
  166. data/docs/momentum.md +0 -19
  167. data/docs/moving_average_convergence_divergence.md +0 -23
  168. data/docs/peaks_and_valleys.md +0 -11
  169. data/docs/relative_strength_index.md +0 -6
  170. data/docs/simple_moving_average.md +0 -8
  171. data/docs/stochastic_oscillator.md +0 -4
  172. data/docs/ta_lib.md +0 -160
  173. data/docs/true_range.md +0 -12
  174. data/lib/patches/dry-cli.rb +0 -228
  175. data/lib/sqa/activity.rb +0 -10
  176. data/lib/sqa/cli.rb +0 -62
  177. data/lib/sqa/commands/analysis.rb +0 -309
  178. data/lib/sqa/commands/base.rb +0 -139
  179. data/lib/sqa/commands/web.rb +0 -199
  180. data/lib/sqa/commands.rb +0 -22
  181. data/lib/sqa/constants.rb +0 -23
  182. data/lib/sqa/indicator/average_true_range.rb +0 -33
  183. data/lib/sqa/indicator/bollinger_bands.rb +0 -28
  184. data/lib/sqa/indicator/candlestick_pattern_recognizer.rb +0 -60
  185. data/lib/sqa/indicator/donchian_channel.rb +0 -29
  186. data/lib/sqa/indicator/double_top_bottom_pattern.rb +0 -34
  187. data/lib/sqa/indicator/elliott_wave_theory.rb +0 -57
  188. data/lib/sqa/indicator/exponential_moving_average.rb +0 -25
  189. data/lib/sqa/indicator/exponential_moving_average_trend.rb +0 -36
  190. data/lib/sqa/indicator/fibonacci_retracement.rb +0 -23
  191. data/lib/sqa/indicator/head_and_shoulders_pattern.rb +0 -26
  192. data/lib/sqa/indicator/market_profile.rb +0 -32
  193. data/lib/sqa/indicator/mean_reversion.rb +0 -37
  194. data/lib/sqa/indicator/momentum.rb +0 -28
  195. data/lib/sqa/indicator/moving_average_convergence_divergence.rb +0 -29
  196. data/lib/sqa/indicator/peaks_and_valleys.rb +0 -29
  197. data/lib/sqa/indicator/predict_next_value.rb +0 -202
  198. data/lib/sqa/indicator/relative_strength_index.rb +0 -47
  199. data/lib/sqa/indicator/simple_moving_average.rb +0 -24
  200. data/lib/sqa/indicator/simple_moving_average_trend.rb +0 -32
  201. data/lib/sqa/indicator/stochastic_oscillator.rb +0 -68
  202. data/lib/sqa/indicator/true_range.rb +0 -39
  203. data/lib/sqa/trade.rb +0 -26
@@ -0,0 +1,723 @@
1
+ /* Reset and Base Styles */
2
+ * {
3
+ margin: 0;
4
+ padding: 0;
5
+ box-sizing: border-box;
6
+ }
7
+
8
+ :root {
9
+ --primary-color: #00d4ff;
10
+ --secondary-color: #00a8cc;
11
+ --success-color: #00ff88;
12
+ --danger-color: #ff3366;
13
+ --warning-color: #ffaa00;
14
+ --dark-bg: #0a0e27;
15
+ --card-bg: #151b3d;
16
+ --card-bg-lighter: #1e2749;
17
+ --text-primary: #e8eaf6;
18
+ --text-secondary: #9fa8da;
19
+ --border-color: #2a3154;
20
+ --positive-color: #00ff88;
21
+ --negative-color: #ff3366;
22
+ --chart-bg: #0f1535;
23
+ --grid-color: #2a3154;
24
+ }
25
+
26
+ body {
27
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, sans-serif;
28
+ background-color: var(--dark-bg);
29
+ color: var(--text-primary);
30
+ line-height: 1.6;
31
+ }
32
+
33
+ /* Navigation */
34
+ .navbar {
35
+ background: linear-gradient(135deg, var(--dark-bg) 0%, #16213e 100%);
36
+ color: white;
37
+ padding: 1rem 0;
38
+ box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
39
+ position: sticky;
40
+ top: 0;
41
+ z-index: 1000;
42
+ }
43
+
44
+ .nav-container {
45
+ max-width: 1200px;
46
+ margin: 0 auto;
47
+ padding: 0 2rem;
48
+ display: flex;
49
+ justify-content: space-between;
50
+ align-items: center;
51
+ }
52
+
53
+ .nav-brand {
54
+ font-size: 1.5rem;
55
+ font-weight: 700;
56
+ display: flex;
57
+ align-items: center;
58
+ gap: 0.5rem;
59
+ }
60
+
61
+ .nav-brand i {
62
+ color: var(--primary-color);
63
+ }
64
+
65
+ .nav-menu {
66
+ display: flex;
67
+ list-style: none;
68
+ gap: 2rem;
69
+ }
70
+
71
+ .nav-link {
72
+ color: white;
73
+ text-decoration: none;
74
+ font-weight: 500;
75
+ transition: color 0.3s ease;
76
+ display: flex;
77
+ align-items: center;
78
+ gap: 0.5rem;
79
+ }
80
+
81
+ .nav-link:hover {
82
+ color: var(--primary-color);
83
+ }
84
+
85
+ /* Main Content */
86
+ .main-content {
87
+ min-height: calc(100vh - 200px);
88
+ padding-bottom: 2rem;
89
+ }
90
+
91
+ /* Hero Section */
92
+ .hero {
93
+ background: linear-gradient(135deg, #1a237e 0%, #4a148c 100%);
94
+ color: white;
95
+ padding: 4rem 2rem;
96
+ text-align: center;
97
+ }
98
+
99
+ .hero-content h1 {
100
+ font-size: 3rem;
101
+ margin-bottom: 1rem;
102
+ font-weight: 700;
103
+ }
104
+
105
+ .hero-subtitle {
106
+ font-size: 1.25rem;
107
+ margin-bottom: 2rem;
108
+ opacity: 0.9;
109
+ }
110
+
111
+ .ticker-search {
112
+ max-width: 600px;
113
+ margin: 2rem auto;
114
+ }
115
+
116
+ .search-form {
117
+ display: flex;
118
+ gap: 1rem;
119
+ }
120
+
121
+ .search-form input {
122
+ flex: 1;
123
+ padding: 1rem;
124
+ border: none;
125
+ border-radius: 8px;
126
+ font-size: 1.1rem;
127
+ outline: none;
128
+ }
129
+
130
+ .quick-links {
131
+ margin-top: 3rem;
132
+ }
133
+
134
+ .quick-links h3 {
135
+ margin-bottom: 1rem;
136
+ font-weight: 600;
137
+ }
138
+
139
+ .ticker-buttons {
140
+ display: flex;
141
+ gap: 1rem;
142
+ justify-content: center;
143
+ flex-wrap: wrap;
144
+ }
145
+
146
+ .ticker-btn {
147
+ background: rgba(255, 255, 255, 0.2);
148
+ color: white;
149
+ padding: 0.75rem 1.5rem;
150
+ border-radius: 8px;
151
+ text-decoration: none;
152
+ font-weight: 600;
153
+ transition: all 0.3s ease;
154
+ backdrop-filter: blur(10px);
155
+ }
156
+
157
+ .ticker-btn:hover {
158
+ background: rgba(255, 255, 255, 0.3);
159
+ transform: translateY(-2px);
160
+ }
161
+
162
+ /* Buttons */
163
+ .btn {
164
+ padding: 0.75rem 1.5rem;
165
+ border: none;
166
+ border-radius: 8px;
167
+ font-size: 1rem;
168
+ font-weight: 600;
169
+ cursor: pointer;
170
+ transition: all 0.3s ease;
171
+ text-decoration: none;
172
+ display: inline-flex;
173
+ align-items: center;
174
+ gap: 0.5rem;
175
+ }
176
+
177
+ .btn-primary {
178
+ background: var(--primary-color);
179
+ color: white;
180
+ }
181
+
182
+ .btn-primary:hover {
183
+ background: var(--secondary-color);
184
+ transform: translateY(-2px);
185
+ box-shadow: 0 4px 12px rgba(33, 150, 243, 0.3);
186
+ }
187
+
188
+ .btn-secondary {
189
+ background: var(--card-bg-lighter);
190
+ color: var(--text-primary);
191
+ border: 2px solid var(--border-color);
192
+ }
193
+
194
+ .btn-secondary:hover {
195
+ border-color: var(--primary-color);
196
+ color: var(--primary-color);
197
+ background: var(--card-bg);
198
+ }
199
+
200
+ .btn-large {
201
+ padding: 1rem 2rem;
202
+ font-size: 1.1rem;
203
+ }
204
+
205
+ .btn-small {
206
+ padding: 0.5rem 1rem;
207
+ font-size: 0.875rem;
208
+ background: var(--card-bg-lighter);
209
+ color: var(--text-primary);
210
+ border: 1px solid var(--border-color);
211
+ }
212
+
213
+ .btn-small.active {
214
+ background: var(--primary-color);
215
+ color: var(--dark-bg);
216
+ border-color: var(--primary-color);
217
+ font-weight: 700;
218
+ }
219
+
220
+ /* Period Selector */
221
+ .period-selector {
222
+ background: var(--card-bg);
223
+ padding: 1.5rem;
224
+ border-radius: 8px;
225
+ margin-bottom: 1.5rem;
226
+ box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
227
+ }
228
+
229
+ .period-selector label {
230
+ font-weight: 600;
231
+ color: var(--text-primary);
232
+ margin-bottom: 0.75rem;
233
+ display: block;
234
+ }
235
+
236
+ .period-buttons {
237
+ display: flex;
238
+ gap: 0.5rem;
239
+ flex-wrap: wrap;
240
+ }
241
+
242
+ .btn-period {
243
+ padding: 0.5rem 1rem;
244
+ font-size: 0.875rem;
245
+ background: var(--card-bg-lighter);
246
+ color: var(--text-primary);
247
+ border: 1px solid var(--border-color);
248
+ border-radius: 4px;
249
+ cursor: pointer;
250
+ transition: all 0.2s;
251
+ font-family: inherit;
252
+ }
253
+
254
+ .btn-period:hover {
255
+ background: var(--card-bg);
256
+ border-color: var(--primary-color);
257
+ color: var(--primary-color);
258
+ }
259
+
260
+ .btn-period.active {
261
+ background: var(--primary-color);
262
+ color: var(--dark-bg);
263
+ border-color: var(--primary-color);
264
+ font-weight: 700;
265
+ }
266
+
267
+ /* Container */
268
+ .container {
269
+ max-width: 1200px;
270
+ margin: 0 auto;
271
+ padding: 0 2rem;
272
+ }
273
+
274
+ /* Features Section */
275
+ .features {
276
+ padding: 4rem 0;
277
+ background: var(--card-bg);
278
+ }
279
+
280
+ .features h2 {
281
+ text-align: center;
282
+ font-size: 2.5rem;
283
+ margin-bottom: 3rem;
284
+ color: var(--text-primary);
285
+ }
286
+
287
+ .feature-grid {
288
+ display: grid;
289
+ grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
290
+ gap: 2rem;
291
+ }
292
+
293
+ .feature-card {
294
+ padding: 2rem;
295
+ background: var(--card-bg-lighter);
296
+ border-radius: 12px;
297
+ border: 1px solid var(--border-color);
298
+ transition: all 0.3s ease;
299
+ }
300
+
301
+ .feature-card:hover {
302
+ transform: translateY(-5px);
303
+ box-shadow: 0 10px 30px rgba(0, 217, 255, 0.2);
304
+ border-color: var(--primary-color);
305
+ }
306
+
307
+ .feature-icon {
308
+ font-size: 3rem;
309
+ color: var(--primary-color);
310
+ margin-bottom: 1rem;
311
+ }
312
+
313
+ .feature-card h3 {
314
+ font-size: 1.5rem;
315
+ margin-bottom: 1rem;
316
+ color: var(--text-primary);
317
+ }
318
+
319
+ .feature-card p {
320
+ color: var(--text-secondary);
321
+ line-height: 1.6;
322
+ }
323
+
324
+ /* Info Section */
325
+ .info-section {
326
+ padding: 4rem 0;
327
+ background: var(--dark-bg);
328
+ }
329
+
330
+ .info-content h2 {
331
+ font-size: 2rem;
332
+ margin-bottom: 1.5rem;
333
+ color: var(--text-primary);
334
+ }
335
+
336
+ .info-content p {
337
+ font-size: 1.1rem;
338
+ color: var(--text-secondary);
339
+ margin-bottom: 1.5rem;
340
+ line-height: 1.8;
341
+ }
342
+
343
+ .info-list {
344
+ list-style: none;
345
+ display: grid;
346
+ grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
347
+ gap: 1rem;
348
+ margin-top: 2rem;
349
+ }
350
+
351
+ .info-list li {
352
+ display: flex;
353
+ align-items: center;
354
+ gap: 0.75rem;
355
+ font-size: 1rem;
356
+ color: var(--text-primary);
357
+ }
358
+
359
+ .info-list i {
360
+ color: var(--success-color);
361
+ font-size: 1.2rem;
362
+ }
363
+
364
+ /* Dashboard */
365
+ .dashboard {
366
+ width: 100%;
367
+ max-width: 100%;
368
+ margin: 0 auto;
369
+ padding: 2rem;
370
+ }
371
+
372
+ .dashboard-header {
373
+ display: flex;
374
+ justify-content: space-between;
375
+ align-items: center;
376
+ margin-bottom: 2rem;
377
+ padding: 1.5rem;
378
+ background: var(--card-bg);
379
+ border-radius: 12px;
380
+ box-shadow: 0 2px 10px rgba(0, 0, 0, 0.3);
381
+ border: 1px solid var(--border-color);
382
+ }
383
+
384
+ .ticker-info h1 {
385
+ font-size: 2.5rem;
386
+ font-weight: 700;
387
+ color: var(--text-primary);
388
+ margin-bottom: 0.5rem;
389
+ }
390
+
391
+ .price-info {
392
+ display: flex;
393
+ gap: 1rem;
394
+ align-items: center;
395
+ }
396
+
397
+ .current-price {
398
+ font-size: 2rem;
399
+ font-weight: 600;
400
+ }
401
+
402
+ .price-change {
403
+ font-size: 1.25rem;
404
+ font-weight: 600;
405
+ padding: 0.25rem 0.75rem;
406
+ border-radius: 6px;
407
+ }
408
+
409
+ .price-change.positive {
410
+ color: var(--positive-color);
411
+ background: rgba(38, 166, 154, 0.1);
412
+ }
413
+
414
+ .price-change.negative {
415
+ color: var(--negative-color);
416
+ background: rgba(239, 83, 80, 0.1);
417
+ }
418
+
419
+ .header-actions {
420
+ display: flex;
421
+ gap: 1rem;
422
+ }
423
+
424
+ /* Metrics Grid */
425
+ .metrics-grid {
426
+ display: grid;
427
+ grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
428
+ gap: 1.5rem;
429
+ margin-bottom: 2rem;
430
+ }
431
+
432
+ .metric-card {
433
+ background: var(--card-bg);
434
+ padding: 1.5rem;
435
+ border-radius: 12px;
436
+ box-shadow: 0 2px 10px rgba(0, 0, 0, 0.3);
437
+ border-left: 4px solid var(--primary-color);
438
+ border: 1px solid var(--border-color);
439
+ border-left: 4px solid var(--primary-color);
440
+ }
441
+
442
+ .metric-label {
443
+ font-size: 0.875rem;
444
+ color: var(--text-secondary);
445
+ margin-bottom: 0.5rem;
446
+ text-transform: uppercase;
447
+ letter-spacing: 0.5px;
448
+ }
449
+
450
+ .metric-value {
451
+ font-size: 1.75rem;
452
+ font-weight: 700;
453
+ color: var(--text-primary);
454
+ }
455
+
456
+ .metric-value.signal-buy {
457
+ color: var(--success-color);
458
+ }
459
+
460
+ .metric-value.signal-sell {
461
+ color: var(--danger-color);
462
+ }
463
+
464
+ .metric-value.signal-neutral {
465
+ color: var(--warning-color);
466
+ }
467
+
468
+ .metric-signal {
469
+ font-size: 0.875rem;
470
+ margin-top: 0.5rem;
471
+ padding: 0.25rem 0.5rem;
472
+ border-radius: 4px;
473
+ display: inline-block;
474
+ }
475
+
476
+ .metric-signal.signal-buy {
477
+ background: rgba(76, 175, 80, 0.1);
478
+ color: var(--success-color);
479
+ }
480
+
481
+ .metric-signal.signal-sell {
482
+ background: rgba(244, 67, 54, 0.1);
483
+ color: var(--danger-color);
484
+ }
485
+
486
+ .metric-signal.signal-neutral {
487
+ background: rgba(255, 152, 0, 0.1);
488
+ color: var(--warning-color);
489
+ }
490
+
491
+ /* Charts */
492
+ .chart-container {
493
+ background: var(--card-bg);
494
+ padding: 1.5rem;
495
+ border-radius: 12px;
496
+ box-shadow: 0 2px 10px rgba(0, 0, 0, 0.3);
497
+ margin-bottom: 2rem;
498
+ border: 1px solid var(--border-color);
499
+ }
500
+
501
+ .chart-header {
502
+ display: flex;
503
+ justify-content: space-between;
504
+ align-items: center;
505
+ margin-bottom: 1rem;
506
+ }
507
+
508
+ .chart-header h2,
509
+ .chart-header h3 {
510
+ display: flex;
511
+ align-items: center;
512
+ gap: 0.5rem;
513
+ color: var(--text-primary);
514
+ }
515
+
516
+ .chart-header i {
517
+ color: var(--primary-color);
518
+ }
519
+
520
+ .chart-controls {
521
+ display: flex;
522
+ gap: 0.5rem;
523
+ }
524
+
525
+ .chart {
526
+ width: 100%;
527
+ height: 500px;
528
+ }
529
+
530
+ .chart-small {
531
+ width: 100%;
532
+ height: 350px;
533
+ }
534
+
535
+ .indicators-grid {
536
+ display: flex;
537
+ flex-direction: column;
538
+ gap: 1.5rem;
539
+ margin-bottom: 2rem;
540
+ }
541
+
542
+ /* Strategy Results */
543
+ .strategy-results {
544
+ padding: 1rem 0;
545
+ }
546
+
547
+ .results-table {
548
+ width: 100%;
549
+ border-collapse: collapse;
550
+ margin-top: 1rem;
551
+ }
552
+
553
+ .results-table thead {
554
+ background: var(--card-bg-lighter);
555
+ }
556
+
557
+ .results-table th,
558
+ .results-table td {
559
+ padding: 1rem;
560
+ text-align: left;
561
+ border-bottom: 1px solid var(--border-color);
562
+ color: var(--text-primary);
563
+ }
564
+
565
+ .results-table th {
566
+ font-weight: 600;
567
+ color: var(--text-secondary);
568
+ text-transform: uppercase;
569
+ font-size: 0.875rem;
570
+ letter-spacing: 0.5px;
571
+ }
572
+
573
+ .results-table td {
574
+ font-weight: 500;
575
+ }
576
+
577
+ .results-table tr:hover {
578
+ background: var(--card-bg-lighter);
579
+ }
580
+
581
+ .results-table .positive {
582
+ color: var(--positive-color);
583
+ font-weight: 600;
584
+ }
585
+
586
+ .results-table .negative {
587
+ color: var(--negative-color);
588
+ font-weight: 600;
589
+ }
590
+
591
+ .hint {
592
+ color: var(--text-secondary);
593
+ font-style: italic;
594
+ padding: 2rem;
595
+ text-align: center;
596
+ }
597
+
598
+ .loading {
599
+ text-align: center;
600
+ padding: 2rem;
601
+ color: var(--primary-color);
602
+ font-size: 1.1rem;
603
+ }
604
+
605
+ .error {
606
+ text-align: center;
607
+ padding: 2rem;
608
+ color: var(--danger-color);
609
+ }
610
+
611
+ /* Modal */
612
+ .modal {
613
+ display: none;
614
+ position: fixed;
615
+ z-index: 2000;
616
+ left: 0;
617
+ top: 0;
618
+ width: 100%;
619
+ height: 100%;
620
+ background-color: rgba(0, 0, 0, 0.5);
621
+ backdrop-filter: blur(5px);
622
+ }
623
+
624
+ .modal-content {
625
+ background-color: var(--card-bg);
626
+ margin: 15% auto;
627
+ padding: 2rem;
628
+ border-radius: 12px;
629
+ width: 90%;
630
+ max-width: 500px;
631
+ box-shadow: 0 10px 40px rgba(0, 0, 0, 0.5);
632
+ border: 1px solid var(--border-color);
633
+ }
634
+
635
+ .modal-content h2 {
636
+ margin-bottom: 1.5rem;
637
+ color: var(--text-primary);
638
+ }
639
+
640
+ .modal-content input {
641
+ width: 100%;
642
+ padding: 1rem;
643
+ border: 2px solid var(--border-color);
644
+ border-radius: 8px;
645
+ font-size: 1.1rem;
646
+ margin-bottom: 1rem;
647
+ outline: none;
648
+ background: var(--card-bg-lighter);
649
+ color: var(--text-primary);
650
+ }
651
+
652
+ .modal-content input:focus {
653
+ border-color: var(--primary-color);
654
+ background: var(--card-bg);
655
+ }
656
+
657
+ .close {
658
+ color: var(--text-secondary);
659
+ float: right;
660
+ font-size: 2rem;
661
+ font-weight: bold;
662
+ cursor: pointer;
663
+ transition: color 0.3s ease;
664
+ }
665
+
666
+ .close:hover {
667
+ color: var(--text-primary);
668
+ }
669
+
670
+ /* Footer */
671
+ .footer {
672
+ background: var(--dark-bg);
673
+ color: white;
674
+ padding: 2rem 0;
675
+ margin-top: 4rem;
676
+ }
677
+
678
+ .footer-content {
679
+ max-width: 1200px;
680
+ margin: 0 auto;
681
+ padding: 0 2rem;
682
+ text-align: center;
683
+ }
684
+
685
+ .footer-content p {
686
+ margin-bottom: 0.5rem;
687
+ opacity: 0.8;
688
+ }
689
+
690
+ .disclaimer {
691
+ color: var(--warning-color);
692
+ font-weight: 600;
693
+ }
694
+
695
+ /* Responsive */
696
+ @media (max-width: 768px) {
697
+ .hero-content h1 {
698
+ font-size: 2rem;
699
+ }
700
+
701
+ .search-form {
702
+ flex-direction: column;
703
+ }
704
+
705
+ .dashboard-header {
706
+ flex-direction: column;
707
+ align-items: flex-start;
708
+ gap: 1rem;
709
+ }
710
+
711
+ .metrics-grid {
712
+ grid-template-columns: 1fr;
713
+ }
714
+
715
+ .indicators-grid {
716
+ grid-template-columns: 1fr;
717
+ }
718
+
719
+ .nav-menu {
720
+ gap: 1rem;
721
+ font-size: 0.875rem;
722
+ }
723
+ }