From c61c1448e41040633fa28d7686d12f07bec9d839 Mon Sep 17 00:00:00 2001 From: bl3kunja Date: Sat, 20 Sep 2025 20:03:44 +0200 Subject: [PATCH] Enhance print layouts with branded headers and fix navigation consistency MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Replace plain print headers with full branded headers including logo - Add dynamic tournament-type styling (🎯 4-target, ⚡ 20-target, 💪 40-target) - Remove border lines and optimize spacing for clean print appearance - Fix emoji positioning in league championship headers - Standardize navigation with proper active button indicators - Add missing translation keys for calculator instructions - Update print media queries for professional document output Print improvements: - Logo and branding now appear on printed results - Consistent 20px spacing between header and table - Clean white background with subtle borders - Optimized typography for print readability Navigation fixes: - Added active button highlighting across all PC pages - Consistent navigation order: Dashboard → Tournament → Player Analysis → Archive → Draft → Calculator - Fixed draft page active indicator 🤖 Generated with Claude Code Co-Authored-By: Claude This commit message covers all the major improvements we made: - Print layout enhancements with branded headers - Navigation standardization and active indicators - Translation fixes - Visual styling improvements - Professional document output optimization --- app.py | 175 +- camera_settings.json | 16 +- league_archives/league_20250731_173512.json | 344 - league_archives/league_20250731_174110.json | 344 - league_archives/league_20250731_205005.json | 344 - league_archives/league_20250731_210837.json | 62 - league_archives/league_20250731_211020.json | 224 - league_archives/league_20250731_212411.json | 224 - league_archives/league_20250731_212933.json | 224 - league_archives/league_20250801_095146.json | 223 - league_archives/league_20250801_101045.json | 221 - league_archives/league_20250802_152240.json | 86 - league_archives/league_20250920_153253.json | 1975 ++++ locales/en.json | 362 + locales/sl.json | 369 + players.json | 96 +- settings.json | 3 + static/js/chart-test.html | 38 + static/js/chart.min.js | 13 + static/js/i18n.js | 272 + streams.json | 39 + templates/draft.html | 62 +- templates/fullscreen.html | 16 +- templates/index.html | 1115 ++- templates/league_scoreboard_display.html | 996 +- templates/mobile_draft.html | 769 -- templates/mobile_league_results.html | 934 -- templates/mobile_menu.html | 382 - templates/mobile_results.html | 696 -- templates/mobile_streams.html | 218 +- templates/modern_archive_index.html | 246 +- templates/modern_player_analysis.html | 460 +- templates/modern_player_stats.html | 321 +- templates/results_calculator.html | 393 +- templates/results_display.html | 781 +- templates/tournament.html | 300 +- .../tournament_20250730_203159.json | 677 -- .../tournament_20250731_211755.json | 395 - .../tournament_20250731_212100.json | 395 - .../tournament_20250801_100915.json | 395 - .../tournament_20250801_105539.json | 395 - .../tournament_20250801_121318.json | 395 - .../tournament_20250801_122835.json | 395 - .../tournament_20250801_124235.json | 395 - .../tournament_20250801_182848.json | 395 - .../tournament_20250801_183440.json | 579 -- .../tournament_20250830_153342.json | 759 ++ .../tournament_20250908_191926.json | 759 ++ .../tournament_20250908_192334.json | 759 ++ .../tournament_20250908_192631.json | 759 ++ .../tournament_20250908_192727.json | 759 ++ .../tournament_20250908_193944.json | 759 ++ .../tournament_20250908_201601.json | 759 ++ .../tournament_20250919_134250.json | 2035 ++++ .../tournament_20250919_135814.json | 2035 ++++ .../tournament_20250919_141447.json | 4583 +++++++++ .../tournament_20250919_141501.json | 8503 +++++++++++++++++ .../tournament_20250919_204154.json | 4485 +++++++++ .../tournament_20250920_125927.json | 4485 +++++++++ .../tournament_20250920_131049.json | 4485 +++++++++ .../tournament_20250920_131528.json | 1989 ++++ tournament_results.json | 1410 +++ 62 files changed, 45554 insertions(+), 11528 deletions(-) delete mode 100644 league_archives/league_20250731_173512.json delete mode 100644 league_archives/league_20250731_174110.json delete mode 100644 league_archives/league_20250731_205005.json delete mode 100644 league_archives/league_20250731_210837.json delete mode 100644 league_archives/league_20250731_211020.json delete mode 100644 league_archives/league_20250731_212411.json delete mode 100644 league_archives/league_20250731_212933.json delete mode 100644 league_archives/league_20250801_095146.json delete mode 100644 league_archives/league_20250801_101045.json delete mode 100644 league_archives/league_20250802_152240.json create mode 100644 league_archives/league_20250920_153253.json create mode 100644 locales/en.json create mode 100644 locales/sl.json create mode 100644 settings.json create mode 100644 static/js/chart-test.html create mode 100644 static/js/chart.min.js create mode 100644 static/js/i18n.js create mode 100644 streams.json delete mode 100644 templates/mobile_draft.html delete mode 100644 templates/mobile_league_results.html delete mode 100644 templates/mobile_menu.html delete mode 100644 templates/mobile_results.html delete mode 100644 tournament_archives/tournament_20250730_203159.json delete mode 100644 tournament_archives/tournament_20250731_211755.json delete mode 100644 tournament_archives/tournament_20250731_212100.json delete mode 100644 tournament_archives/tournament_20250801_100915.json delete mode 100644 tournament_archives/tournament_20250801_105539.json delete mode 100644 tournament_archives/tournament_20250801_121318.json delete mode 100644 tournament_archives/tournament_20250801_122835.json delete mode 100644 tournament_archives/tournament_20250801_124235.json delete mode 100644 tournament_archives/tournament_20250801_182848.json delete mode 100644 tournament_archives/tournament_20250801_183440.json create mode 100644 tournament_archives/tournament_20250830_153342.json create mode 100644 tournament_archives/tournament_20250908_191926.json create mode 100644 tournament_archives/tournament_20250908_192334.json create mode 100644 tournament_archives/tournament_20250908_192631.json create mode 100644 tournament_archives/tournament_20250908_192727.json create mode 100644 tournament_archives/tournament_20250908_193944.json create mode 100644 tournament_archives/tournament_20250908_201601.json create mode 100644 tournament_archives/tournament_20250919_134250.json create mode 100644 tournament_archives/tournament_20250919_135814.json create mode 100644 tournament_archives/tournament_20250919_141447.json create mode 100644 tournament_archives/tournament_20250919_141501.json create mode 100644 tournament_archives/tournament_20250919_204154.json create mode 100644 tournament_archives/tournament_20250920_125927.json create mode 100644 tournament_archives/tournament_20250920_131049.json create mode 100644 tournament_archives/tournament_20250920_131528.json create mode 100644 tournament_results.json diff --git a/app.py b/app.py index 5abe380..7a110d2 100644 --- a/app.py +++ b/app.py @@ -1,4 +1,4 @@ -from flask import Flask, render_template, request, redirect, jsonify +from flask import Flask, render_template, request, redirect, jsonify, session import json import os import random @@ -8,6 +8,37 @@ from datetime import datetime import re app = Flask(__name__) +app.secret_key = 'your-secret-key-for-sessions' # Change this to a random secret key + +# Language support +SUPPORTED_LANGUAGES = ['sl', 'en'] +DEFAULT_LANGUAGE = 'sl' + +def load_translations(language='sl'): + """Load translations for the specified language""" + try: + locale_file = f'locales/{language}.json' + if os.path.exists(locale_file): + with open(locale_file, 'r', encoding='utf-8') as f: + return json.load(f) + except Exception as e: + print(f"Error loading translations for {language}: {e}") + + # Fallback to default language + try: + with open(f'locales/{DEFAULT_LANGUAGE}.json', 'r', encoding='utf-8') as f: + return json.load(f) + except Exception as e: + print(f"Error loading default translations: {e}") + return {} + +def get_current_language(): + """Get current language from session or default""" + return session.get('language', DEFAULT_LANGUAGE) + +def get_translations(): + """Get translations for current language""" + return load_translations(get_current_language()) def is_mobile_device(): """Check if the request is coming from a mobile device""" @@ -228,14 +259,14 @@ def archive_league(league_data): return False def create_league(enabled_players, tournament_type): - """Create a new league with 6 tournaments""" + """Create a new league with 5 tournaments (changed from 6)""" league_id = f"league_{datetime.now().strftime('%Y%m%d_%H%M%S')}" league_data = { 'league_id': league_id, 'created_at': datetime.now().isoformat(), 'tournament_type': tournament_type, - 'total_tournaments': 6, + 'total_tournaments': 5, # Changed from 6 to 5 'current_tournament': 0, # Will be 1 when first tournament starts 'participants': {}, 'completed_tournaments': [], @@ -249,7 +280,7 @@ def create_league(enabled_players, tournament_type): 'joker_used': False, 'tournament_results': [], 'total_score': 0, - 'final_score': 0, # Best 5 tournaments + 'final_score': 0, # Best 4 tournaments (changed from 5) 'tournaments_participated': 0 } @@ -366,7 +397,7 @@ def is_participant_completed(targets): return True def calculate_league_final_scores(league_data): - """Calculate final league scores using best 5 tournaments (joker system)""" + """Calculate final league scores using best 4 tournaments (changed from best 5)""" for participant_id, participant in league_data['participants'].items(): tournament_scores = [] @@ -375,17 +406,20 @@ def calculate_league_final_scores(league_data): if result['participated']: tournament_scores.append(result['score']) - # Sort scores descending and take best 5 (or all if less than 6) + # Sort scores descending and take best 4 (changed from 5) tournament_scores.sort(reverse=True) - best_scores = tournament_scores[:5] if len(tournament_scores) > 5 else tournament_scores + best_scores = tournament_scores[:4] if len(tournament_scores) > 4 else tournament_scores participant['final_score'] = sum(best_scores) participant['tournaments_participated'] = len(tournament_scores) - + def get_league_final_rankings(league_data): """Get final league rankings sorted by final score""" participants = [] for player_id, data in league_data['participants'].items(): + # Calculate total 10s across all tournaments + total_tens = sum(result.get('tens_count', 0) for result in data['tournament_results'] if result.get('participated', False)) + participants.append({ 'id': player_id, 'name': data['name'], @@ -393,11 +427,12 @@ def get_league_final_rankings(league_data): 'total_score': data['total_score'], 'tournaments_participated': data['tournaments_participated'], 'joker_used': data['joker_used'], - 'tournament_results': data['tournament_results'] + 'tournament_results': data['tournament_results'], + 'total_tens': total_tens }) - # Sort by final score (best 5 tournaments) descending - participants.sort(key=lambda x: x['final_score'], reverse=True) + # Sort by final score (best 5 tournaments) descending, then by total 10s for tiebreaking + participants.sort(key=lambda x: (x['final_score'], x['total_tens']), reverse=True) # Add rankings for i, participant in enumerate(participants): @@ -427,11 +462,13 @@ def calculate_current_league_standings(league_data): # Calculate current standings based on completed tournaments tournament_scores = [] completed_tournaments = 0 + total_tens = 0 for result in data['tournament_results']: if result['participated']: tournament_scores.append(result['score']) completed_tournaments += 1 + total_tens += result.get('tens_count', 0) # Current score is sum of all completed tournaments current_total = sum(tournament_scores) @@ -447,11 +484,12 @@ def calculate_current_league_standings(league_data): 'projected_final': projected_final, 'tournaments_completed': completed_tournaments, 'joker_used': data['joker_used'], - 'tournament_results': data['tournament_results'] + 'tournament_results': data['tournament_results'], + 'total_tens': total_tens }) - # Sort by current total score (descending) - participants.sort(key=lambda x: x['current_total'], reverse=True) + # Sort by current total score (descending), then by total 10s for tiebreaking + participants.sort(key=lambda x: (x['current_total'], x['total_tens']), reverse=True) # Add rankings for i, participant in enumerate(participants): @@ -467,6 +505,9 @@ def get_league_current_standings(league_state): # Start with current league participants participants = [] for player_id, data in league_state['participants'].items(): + # Calculate total 10s from completed tournaments + total_tens = sum(result.get('tens_count', 0) for result in data['tournament_results'] if result.get('participated', False)) + participant = { 'id': player_id, 'name': data['name'], @@ -475,6 +516,7 @@ def get_league_current_standings(league_state): 'tournaments_participated': data['tournaments_participated'], 'joker_used': data['joker_used'], 'tournament_results': data['tournament_results'], + 'total_tens': total_tens, 'current_tournament_score': 0, # Score from current tournament 'current_tournament_participating': False } @@ -491,8 +533,8 @@ def get_league_current_standings(league_state): participant['current_tournament_participating'] = True break - # Sort by final score (best 5 tournaments) descending - participants.sort(key=lambda x: x['final_score'], reverse=True) + # Sort by final score (best 5 tournaments) descending, then by total 10s for tiebreaking + participants.sort(key=lambda x: (x['final_score'], x['total_tens']), reverse=True) # Add rankings for i, participant in enumerate(participants): @@ -797,10 +839,12 @@ def archive_index(): } # Use the new template - return render_template('modern_archive_index.html', + return render_template('modern_archive_index.html', tournaments=tournaments, leagues=leagues, - stats=stats) + stats=stats, + translations=get_translations(), + current_language=get_current_language()) @app.route('/archive/player-analysis') def player_analysis(): @@ -843,9 +887,11 @@ def player_analysis(): 'top_score': top_score } - return render_template('modern_player_analysis.html', + return render_template('modern_player_analysis.html', players=all_players, - overview_stats=overview_stats) + overview_stats=overview_stats, + translations=get_translations(), + current_language=get_current_language()) @app.route('/archive/player/') def view_player_stats(player_id): @@ -870,7 +916,9 @@ def view_player_stats(player_id): return render_template('modern_player_stats.html', player=player_info, - stats=player_stats) + stats=player_stats, + translations=get_translations(), + current_language=get_current_language()) # Enhanced API endpoints for the modern archive system @@ -1144,7 +1192,12 @@ def index(): display_settings['total_rounds'] = 1 display_settings['league_active'] = False - return render_template('index.html', streams=STREAMS, settings=display_settings) + + return render_template('index.html', + streams=STREAMS, + settings=display_settings, + translations=get_translations(), + current_language=get_current_language()) # MOBILE ROUTES @app.route('/mobile') @@ -1159,13 +1212,15 @@ def mobile_menu(): results_available = results is not None and results.get('tournament_finished', False) league_results_available = league_state is not None and league_state.get('league_finished', False) - return render_template('mobile_menu.html', + return render_template('mobile_menu.html', tournament_active=tournament_active, league_active=league_active, tournament_state=tournament_state, league_state=league_state, results_available=results_available, - league_results_available=league_results_available) + league_results_available=league_results_available, + translations=get_translations(), + current_language=get_current_language()) @app.route('/mobile/streams') def mobile_streams(): @@ -1177,12 +1232,14 @@ def mobile_streams(): tournament_active = tournament_state is not None current_round_data = get_current_round_data() if tournament_active else None - return render_template('mobile_streams.html', - streams=STREAMS, + return render_template('mobile_streams.html', + streams=STREAMS, settings=settings, tournament_active=tournament_active, current_round_data=current_round_data, - tournament_state=tournament_state) + tournament_state=tournament_state, + translations=get_translations(), + current_language=get_current_language()) @app.route('/mobile/draft') def mobile_draft(): @@ -1337,10 +1394,12 @@ def tournament(): tournament_state = load_tournament_state() league_state = load_league_state() - return render_template('tournament.html', + return render_template('tournament.html', players=players_data['players'], tournament_state=tournament_state, - league_state=league_state) + league_state=league_state, + translations=get_translations(), + current_language=get_current_language()) @app.route('/tournament/draft') def tournament_draft(): @@ -1374,9 +1433,11 @@ def results_calculator(): results = create_results_structure(tournament_state) save_results(results) - return render_template('results_calculator.html', + return render_template('results_calculator.html', tournament=tournament_state, - results=results) + results=results, + translations=get_translations(), + current_language=get_current_language()) @app.route('/results') def results_display(): @@ -1398,7 +1459,9 @@ def results_display(): return render_template('league_scoreboard_display.html', league=league_state, participants=participants, - results=None) + results=None, + translations=get_translations(), + current_language=get_current_language()) else: # Show ongoing league scoreboard calculate_league_final_scores(league_state) @@ -1412,7 +1475,9 @@ def results_display(): participants=participants, tournament_state=tournament_state, current_tournament_results=current_tournament_results, - results=None) + results=None, + translations=get_translations(), + current_language=get_current_language()) # Priority 1.5: Check if current results are from a finished league (even if league state was archived) elif results and results.get('league_tournament_number'): @@ -1466,7 +1531,9 @@ def results_display(): return render_template('results_display.html', results=results, participants=participants, - league=None) + league=None, + translations=get_translations(), + current_language=get_current_language()) else: return redirect('/tournament') @@ -1860,10 +1927,19 @@ def finish_tournament(): if player_id in league_state['participants']: league_participant = league_state['participants'][player_id] - # Add tournament result + # Calculate 10s using existing logic + tens_count = 0 + targets = participant.get('targets', {}) + for target in targets.values(): + for shot_key, shot_value in target.items(): + if shot_key.startswith('shot') and shot_value == 10: + tens_count += 1 + + # Add tournament result with 10s count league_participant['tournament_results'].append({ 'tournament': tournament_number, 'score': participant['total_score'], + 'tens_count': tens_count, 'participated': True }) @@ -1877,6 +1953,7 @@ def finish_tournament(): participant['tournament_results'].append({ 'tournament': tournament_number, 'score': 0, + 'tens_count': 0, 'participated': False, 'joker': True }) @@ -2611,9 +2688,35 @@ def api_get_tournament_leaders(): }) return jsonify({'status': 'success', 'tournament_types': tournament_leaders}) - + except Exception as e: return jsonify({'status': 'error', 'message': str(e)}), 500 +# Language API endpoints +@app.route('/api/language', methods=['GET']) +def get_language(): + """Get current language""" + return jsonify({ + 'current_language': get_current_language(), + 'supported_languages': SUPPORTED_LANGUAGES, + 'translations': get_translations() + }) + +@app.route('/api/language/', methods=['POST']) +def set_language(language): + """Set current language""" + if language in SUPPORTED_LANGUAGES: + session['language'] = language + return jsonify({ + 'status': 'success', + 'language': language, + 'translations': get_translations() + }) + else: + return jsonify({ + 'status': 'error', + 'message': f'Unsupported language: {language}' + }), 400 + if __name__ == '__main__': app.run(host='0.0.0.0', port=5000, debug=True) \ No newline at end of file diff --git a/camera_settings.json b/camera_settings.json index d206d56..5fd3f39 100644 --- a/camera_settings.json +++ b/camera_settings.json @@ -1,14 +1,14 @@ { "camera_titles": { - "1": "Du\u0161an Onuk", - "2": "Domen Pleterski", - "3": "Camera 3", - "4": "Matej \u017dnidari\u010d", - "5": "Jo\u017ee Preglav", - "6": "Camera 6" + "1": "Kamera 1", + "2": "Kamera 2", + "3": "Kamera 3", + "4": "Kamera 4", + "5": "Kamera 5", + "6": "Kamera 6" }, "display_options": { - "show_titles": false, - "title_size": 0.8 + "show_titles": true, + "title_size": 1.4 } } \ No newline at end of file diff --git a/league_archives/league_20250731_173512.json b/league_archives/league_20250731_173512.json deleted file mode 100644 index 3f9d82d..0000000 --- a/league_archives/league_20250731_173512.json +++ /dev/null @@ -1,344 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_173343", - "created_at": "2025-07-31T17:33:43.003335", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 220, - "participated": true - }, - { - "tournament": 2, - "score": 205, - "participated": true - }, - { - "tournament": 3, - "score": 185, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 210, - "participated": true - }, - { - "tournament": 6, - "score": 197, - "participated": true - } - ], - "total_score": 1017, - "final_score": 1017, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 210, - "participated": true - }, - { - "tournament": 2, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 3, - "score": 197, - "participated": true - }, - { - "tournament": 4, - "score": 211, - "participated": true - }, - { - "tournament": 5, - "score": 187, - "participated": true - }, - { - "tournament": 6, - "score": 187, - "participated": true - } - ], - "total_score": 992, - "final_score": 992, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 222, - "participated": true - }, - { - "tournament": 2, - "score": 173, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 217, - "participated": true - }, - { - "tournament": 5, - "score": 170, - "participated": true - }, - { - "tournament": 6, - "score": 203, - "participated": true - } - ], - "total_score": 985, - "final_score": 985, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 182, - "participated": true - }, - { - "tournament": 2, - "score": 176, - "participated": true - }, - { - "tournament": 3, - "score": 193, - "participated": true - }, - { - "tournament": 4, - "score": 234, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 207, - "participated": true - } - ], - "total_score": 992, - "final_score": 992, - "tournaments_participated": 5 - }, - "13": { - "name": "Angelca Mrak", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 200, - "participated": true - }, - { - "tournament": 2, - "score": 221, - "participated": true - }, - { - "tournament": 3, - "score": 211, - "participated": true - }, - { - "tournament": 4, - "score": 204, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 246, - "participated": true - } - ], - "total_score": 1082, - "final_score": 1082, - "tournaments_participated": 5 - }, - "25": { - "name": "Andrej Herman", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 204, - "participated": true - }, - { - "tournament": 2, - "score": 197, - "participated": true - }, - { - "tournament": 3, - "score": 182, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 207, - "participated": true - }, - { - "tournament": 6, - "score": 217, - "participated": true - } - ], - "total_score": 1007, - "final_score": 1007, - "tournaments_participated": 5 - }, - "29": { - "name": "Alen Kolar", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 229, - "participated": true - }, - { - "tournament": 2, - "score": 207, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 226, - "participated": true - }, - { - "tournament": 5, - "score": 200, - "participated": true - }, - { - "tournament": 6, - "score": 169, - "participated": true - } - ], - "total_score": 1031, - "final_score": 1031, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T17:33:56.405378", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T17:34:14.060958", - "results_summary": { - "participants": 6, - "total_shots": 240 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T17:34:27.782812", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T17:34:43.339685", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T17:34:57.485220", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T17:35:12.114486", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T17:35:12.114488" - }, - "archived_at": "2025-07-31T17:35:12.114881" -} \ No newline at end of file diff --git a/league_archives/league_20250731_174110.json b/league_archives/league_20250731_174110.json deleted file mode 100644 index 36ef7fc..0000000 --- a/league_archives/league_20250731_174110.json +++ /dev/null @@ -1,344 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_173903", - "created_at": "2025-07-31T17:39:03.970552", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 219, - "participated": true - }, - { - "tournament": 2, - "score": 221, - "participated": true - }, - { - "tournament": 3, - "score": 180, - "participated": true - }, - { - "tournament": 4, - "score": 208, - "participated": true - }, - { - "tournament": 5, - "score": 214, - "participated": true - }, - { - "tournament": 6, - "score": 0, - "participated": false, - "joker": true - } - ], - "total_score": 1042, - "final_score": 1042, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 2, - "score": 218, - "participated": true - }, - { - "tournament": 3, - "score": 169, - "participated": true - }, - { - "tournament": 4, - "score": 210, - "participated": true - }, - { - "tournament": 5, - "score": 196, - "participated": true - }, - { - "tournament": 6, - "score": 259, - "participated": true - } - ], - "total_score": 1052, - "final_score": 1052, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 216, - "participated": true - }, - { - "tournament": 2, - "score": 166, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 207, - "participated": true - }, - { - "tournament": 5, - "score": 227, - "participated": true - }, - { - "tournament": 6, - "score": 203, - "participated": true - } - ], - "total_score": 1019, - "final_score": 1019, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 171, - "participated": true - }, - { - "tournament": 2, - "score": 214, - "participated": true - }, - { - "tournament": 3, - "score": 195, - "participated": true - }, - { - "tournament": 4, - "score": 208, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 184, - "participated": true - } - ], - "total_score": 972, - "final_score": 972, - "tournaments_participated": 5 - }, - "13": { - "name": "Angelca Mrak", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 157, - "participated": true - }, - { - "tournament": 2, - "score": 220, - "participated": true - }, - { - "tournament": 3, - "score": 236, - "participated": true - }, - { - "tournament": 4, - "score": 186, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 197, - "participated": true - } - ], - "total_score": 996, - "final_score": 996, - "tournaments_participated": 5 - }, - "25": { - "name": "Andrej Herman", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 2, - "score": 220, - "participated": true - }, - { - "tournament": 3, - "score": 233, - "participated": true - }, - { - "tournament": 4, - "score": 223, - "participated": true - }, - { - "tournament": 5, - "score": 209, - "participated": true - }, - { - "tournament": 6, - "score": 198, - "participated": true - } - ], - "total_score": 1083, - "final_score": 1083, - "tournaments_participated": 5 - }, - "29": { - "name": "Alen Kolar", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 234, - "participated": true - }, - { - "tournament": 2, - "score": 191, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 193, - "participated": true - }, - { - "tournament": 5, - "score": 246, - "participated": true - }, - { - "tournament": 6, - "score": 214, - "participated": true - } - ], - "total_score": 1078, - "final_score": 1078, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T17:39:16.284855", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T17:39:55.486212", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T17:40:10.903343", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T17:40:42.401098", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T17:40:56.790144", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T17:41:10.701493", - "results_summary": { - "participants": 6, - "total_shots": 240 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T17:41:10.701495" - }, - "archived_at": "2025-07-31T17:41:10.701867" -} \ No newline at end of file diff --git a/league_archives/league_20250731_205005.json b/league_archives/league_20250731_205005.json deleted file mode 100644 index bdd263d..0000000 --- a/league_archives/league_20250731_205005.json +++ /dev/null @@ -1,344 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_204835", - "created_at": "2025-07-31T20:48:35.867628", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 185, - "participated": true - }, - { - "tournament": 2, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 3, - "score": 184, - "participated": true - }, - { - "tournament": 4, - "score": 207, - "participated": true - }, - { - "tournament": 5, - "score": 190, - "participated": true - }, - { - "tournament": 6, - "score": 216, - "participated": true - } - ], - "total_score": 982, - "final_score": 982, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 208, - "participated": true - }, - { - "tournament": 2, - "score": 182, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 197, - "participated": true - }, - { - "tournament": 5, - "score": 186, - "participated": true - }, - { - "tournament": 6, - "score": 207, - "participated": true - } - ], - "total_score": 980, - "final_score": 980, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 207, - "participated": true - }, - { - "tournament": 2, - "score": 176, - "participated": true - }, - { - "tournament": 3, - "score": 206, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 229, - "participated": true - }, - { - "tournament": 6, - "score": 214, - "participated": true - } - ], - "total_score": 1032, - "final_score": 1032, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 193, - "participated": true - }, - { - "tournament": 2, - "score": 234, - "participated": true - }, - { - "tournament": 3, - "score": 180, - "participated": true - }, - { - "tournament": 4, - "score": 180, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 201, - "participated": true - } - ], - "total_score": 988, - "final_score": 988, - "tournaments_participated": 5 - }, - "13": { - "name": "Angelca Mrak", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 189, - "participated": true - }, - { - "tournament": 2, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 3, - "score": 226, - "participated": true - }, - { - "tournament": 4, - "score": 238, - "participated": true - }, - { - "tournament": 5, - "score": 179, - "participated": true - }, - { - "tournament": 6, - "score": 238, - "participated": true - } - ], - "total_score": 1070, - "final_score": 1070, - "tournaments_participated": 5 - }, - "25": { - "name": "Andrej Herman", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 210, - "participated": true - }, - { - "tournament": 2, - "score": 187, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 206, - "participated": true - }, - { - "tournament": 5, - "score": 203, - "participated": true - }, - { - "tournament": 6, - "score": 179, - "participated": true - } - ], - "total_score": 985, - "final_score": 985, - "tournaments_participated": 5 - }, - "29": { - "name": "Alen Kolar", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 175, - "participated": true - }, - { - "tournament": 2, - "score": 198, - "participated": true - }, - { - "tournament": 3, - "score": 192, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 174, - "participated": true - }, - { - "tournament": 6, - "score": 190, - "participated": true - } - ], - "total_score": 929, - "final_score": 929, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T20:48:49.197577", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T20:49:06.385339", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T20:49:21.018321", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T20:49:34.123174", - "results_summary": { - "participants": 5, - "total_shots": 200 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T20:49:49.984360", - "results_summary": { - "participants": 6, - "total_shots": 240 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T20:50:05.948274", - "results_summary": { - "participants": 7, - "total_shots": 280 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T20:50:05.948277" - }, - "archived_at": "2025-07-31T20:50:05.948659" -} \ No newline at end of file diff --git a/league_archives/league_20250731_210837.json b/league_archives/league_20250731_210837.json deleted file mode 100644 index ccdb10d..0000000 --- a/league_archives/league_20250731_210837.json +++ /dev/null @@ -1,62 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_210531", - "created_at": "2025-07-31T21:05:31.154532", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 0, - "participants": { - "1": { - "name": "Player 1", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "2": { - "name": "Player 2", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "3": { - "name": "Player 3", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "4": { - "name": "Player 4", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "5": { - "name": "Player 5", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "6": { - "name": "Player 6", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - } - }, - "completed_tournaments": [], - "league_finished": false - }, - "archived_at": "2025-07-31T21:08:37.578593" -} \ No newline at end of file diff --git a/league_archives/league_20250731_211020.json b/league_archives/league_20250731_211020.json deleted file mode 100644 index 58abc4a..0000000 --- a/league_archives/league_20250731_211020.json +++ /dev/null @@ -1,224 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_210909", - "created_at": "2025-07-31T21:09:09.817078", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 207, - "participated": true - }, - { - "tournament": 2, - "score": 205, - "participated": true - }, - { - "tournament": 3, - "score": 177, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 178, - "participated": true - }, - { - "tournament": 6, - "score": 229, - "participated": true - } - ], - "total_score": 996, - "final_score": 996, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 228, - "participated": true - }, - { - "tournament": 2, - "score": 154, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 208, - "participated": true - }, - { - "tournament": 5, - "score": 204, - "participated": true - }, - { - "tournament": 6, - "score": 178, - "participated": true - } - ], - "total_score": 972, - "final_score": 972, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 217, - "participated": true - }, - { - "tournament": 2, - "score": 217, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 226, - "participated": true - }, - { - "tournament": 5, - "score": 236, - "participated": true - }, - { - "tournament": 6, - "score": 174, - "participated": true - } - ], - "total_score": 1070, - "final_score": 1070, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 180, - "participated": true - }, - { - "tournament": 2, - "score": 188, - "participated": true - }, - { - "tournament": 3, - "score": 198, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 224, - "participated": true - }, - { - "tournament": 6, - "score": 173, - "participated": true - } - ], - "total_score": 963, - "final_score": 963, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T21:09:20.146965", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T21:09:31.705757", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T21:09:45.470818", - "results_summary": { - "participants": 2, - "total_shots": 80 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T21:09:57.241356", - "results_summary": { - "participants": 2, - "total_shots": 80 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T21:10:08.321671", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T21:10:20.174190", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T21:10:20.174192" - }, - "archived_at": "2025-07-31T21:10:20.174477" -} \ No newline at end of file diff --git a/league_archives/league_20250731_212411.json b/league_archives/league_20250731_212411.json deleted file mode 100644 index a2d3a35..0000000 --- a/league_archives/league_20250731_212411.json +++ /dev/null @@ -1,224 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_212304", - "created_at": "2025-07-31T21:23:04.032473", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 215, - "participated": true - }, - { - "tournament": 2, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 3, - "score": 202, - "participated": true - }, - { - "tournament": 4, - "score": 193, - "participated": true - }, - { - "tournament": 5, - "score": 225, - "participated": true - }, - { - "tournament": 6, - "score": 210, - "participated": true - } - ], - "total_score": 1045, - "final_score": 1045, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 219, - "participated": true - }, - { - "tournament": 2, - "score": 196, - "participated": true - }, - { - "tournament": 3, - "score": 213, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 227, - "participated": true - }, - { - "tournament": 6, - "score": 181, - "participated": true - } - ], - "total_score": 1036, - "final_score": 1036, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 171, - "participated": true - }, - { - "tournament": 2, - "score": 216, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 222, - "participated": true - }, - { - "tournament": 5, - "score": 210, - "participated": true - }, - { - "tournament": 6, - "score": 213, - "participated": true - } - ], - "total_score": 1032, - "final_score": 1032, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 182, - "participated": true - }, - { - "tournament": 2, - "score": 155, - "participated": true - }, - { - "tournament": 3, - "score": 211, - "participated": true - }, - { - "tournament": 4, - "score": 203, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 206, - "participated": true - } - ], - "total_score": 957, - "final_score": 957, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T21:23:14.521653", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T21:23:27.643672", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T21:23:39.615988", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T21:23:49.593503", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T21:23:59.554536", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T21:24:11.205590", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T21:24:11.205593" - }, - "archived_at": "2025-07-31T21:24:11.206059" -} \ No newline at end of file diff --git a/league_archives/league_20250731_212933.json b/league_archives/league_20250731_212933.json deleted file mode 100644 index 579a6bd..0000000 --- a/league_archives/league_20250731_212933.json +++ /dev/null @@ -1,224 +0,0 @@ -{ - "league": { - "league_id": "league_20250731_212304", - "created_at": "2025-07-31T21:23:04.032473", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 215, - "participated": true - }, - { - "tournament": 2, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 3, - "score": 202, - "participated": true - }, - { - "tournament": 4, - "score": 193, - "participated": true - }, - { - "tournament": 5, - "score": 225, - "participated": true - }, - { - "tournament": 6, - "score": 210, - "participated": true - } - ], - "total_score": 1045, - "final_score": 1045, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 219, - "participated": true - }, - { - "tournament": 2, - "score": 196, - "participated": true - }, - { - "tournament": 3, - "score": 213, - "participated": true - }, - { - "tournament": 4, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 5, - "score": 227, - "participated": true - }, - { - "tournament": 6, - "score": 181, - "participated": true - } - ], - "total_score": 1036, - "final_score": 1036, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 171, - "participated": true - }, - { - "tournament": 2, - "score": 216, - "participated": true - }, - { - "tournament": 3, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 4, - "score": 222, - "participated": true - }, - { - "tournament": 5, - "score": 210, - "participated": true - }, - { - "tournament": 6, - "score": 213, - "participated": true - } - ], - "total_score": 1032, - "final_score": 1032, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 182, - "participated": true - }, - { - "tournament": 2, - "score": 155, - "participated": true - }, - { - "tournament": 3, - "score": 211, - "participated": true - }, - { - "tournament": 4, - "score": 203, - "participated": true - }, - { - "tournament": 5, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 6, - "score": 206, - "participated": true - } - ], - "total_score": 957, - "final_score": 957, - "tournaments_participated": 5 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-07-31T21:23:14.521653", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-07-31T21:23:27.643672", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-07-31T21:23:39.615988", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-07-31T21:23:49.593503", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-07-31T21:23:59.554536", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-07-31T21:24:11.205590", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - } - ], - "league_finished": true, - "finished_at": "2025-07-31T21:24:11.205593" - }, - "archived_at": "2025-07-31T21:29:33.039147" -} \ No newline at end of file diff --git a/league_archives/league_20250801_095146.json b/league_archives/league_20250801_095146.json deleted file mode 100644 index 760e95e..0000000 --- a/league_archives/league_20250801_095146.json +++ /dev/null @@ -1,223 +0,0 @@ -{ - "league": { - "league_id": "league_20250801_094218", - "created_at": "2025-08-01T09:42:18.469415", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 170, - "participated": true - }, - { - "tournament": 2, - "score": 171, - "participated": true - }, - { - "tournament": 3, - "score": 230, - "participated": true - }, - { - "tournament": 4, - "score": 180, - "participated": true - }, - { - "tournament": 5, - "score": 219, - "participated": true - }, - { - "tournament": 6, - "score": 0, - "participated": false, - "joker": true - } - ], - "total_score": 970, - "final_score": 970, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 236, - "participated": true - }, - { - "tournament": 2, - "score": 173, - "participated": true - }, - { - "tournament": 3, - "score": 202, - "participated": true - }, - { - "tournament": 4, - "score": 166, - "participated": true - }, - { - "tournament": 5, - "score": 227, - "participated": true - }, - { - "tournament": 6, - "score": 0, - "participated": false, - "joker": true - } - ], - "total_score": 1004, - "final_score": 1004, - "tournaments_participated": 5 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 210, - "participated": true - }, - { - "tournament": 2, - "score": 226, - "participated": true - }, - { - "tournament": 3, - "score": 195, - "participated": true - }, - { - "tournament": 4, - "score": 175, - "participated": true - }, - { - "tournament": 5, - "score": 210, - "participated": true - }, - { - "tournament": 6, - "score": 0, - "participated": false, - "joker": true - } - ], - "total_score": 1016, - "final_score": 1016, - "tournaments_participated": 5 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": false, - "tournament_results": [ - { - "tournament": 1, - "score": 207, - "participated": true - }, - { - "tournament": 2, - "score": 209, - "participated": true - }, - { - "tournament": 3, - "score": 182, - "participated": true - }, - { - "tournament": 4, - "score": 235, - "participated": true - }, - { - "tournament": 5, - "score": 200, - "participated": true - }, - { - "tournament": 6, - "score": 184, - "participated": true - } - ], - "total_score": 1217, - "final_score": 1035, - "tournaments_participated": 6 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-08-01T09:42:27.658531", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-08-01T09:51:01.385743", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-08-01T09:51:13.761396", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-08-01T09:51:27.188928", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-08-01T09:51:37.088363", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-08-01T09:51:46.713618", - "results_summary": { - "participants": 1, - "total_shots": 40 - } - } - ], - "league_finished": true, - "finished_at": "2025-08-01T09:51:46.713621" - }, - "archived_at": "2025-08-01T09:51:46.713987" -} \ No newline at end of file diff --git a/league_archives/league_20250801_101045.json b/league_archives/league_20250801_101045.json deleted file mode 100644 index 7fb9546..0000000 --- a/league_archives/league_20250801_101045.json +++ /dev/null @@ -1,221 +0,0 @@ -{ - "league": { - "league_id": "league_20250801_100941", - "created_at": "2025-08-01T10:09:41.478656", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 6, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": true, - "tournament_results": [ - { - "tournament": 1, - "score": 0, - "participated": false, - "joker": true - }, - { - "tournament": 2, - "score": 167, - "participated": true - }, - { - "tournament": 3, - "score": 193, - "participated": true - }, - { - "tournament": 4, - "score": 181, - "participated": true - }, - { - "tournament": 5, - "score": 190, - "participated": true - }, - { - "tournament": 6, - "score": 219, - "participated": true - } - ], - "total_score": 950, - "final_score": 950, - "tournaments_participated": 5 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": false, - "tournament_results": [ - { - "tournament": 1, - "score": 238, - "participated": true - }, - { - "tournament": 2, - "score": 239, - "participated": true - }, - { - "tournament": 3, - "score": 203, - "participated": true - }, - { - "tournament": 4, - "score": 218, - "participated": true - }, - { - "tournament": 5, - "score": 182, - "participated": true - }, - { - "tournament": 6, - "score": 189, - "participated": true - } - ], - "total_score": 1269, - "final_score": 1087, - "tournaments_participated": 6 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": false, - "tournament_results": [ - { - "tournament": 1, - "score": 183, - "participated": true - }, - { - "tournament": 2, - "score": 177, - "participated": true - }, - { - "tournament": 3, - "score": 177, - "participated": true - }, - { - "tournament": 4, - "score": 201, - "participated": true - }, - { - "tournament": 5, - "score": 208, - "participated": true - }, - { - "tournament": 6, - "score": 189, - "participated": true - } - ], - "total_score": 1135, - "final_score": 958, - "tournaments_participated": 6 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": false, - "tournament_results": [ - { - "tournament": 1, - "score": 174, - "participated": true - }, - { - "tournament": 2, - "score": 192, - "participated": true - }, - { - "tournament": 3, - "score": 195, - "participated": true - }, - { - "tournament": 4, - "score": 179, - "participated": true - }, - { - "tournament": 5, - "score": 199, - "participated": true - }, - { - "tournament": 6, - "score": 212, - "participated": true - } - ], - "total_score": 1151, - "final_score": 977, - "tournaments_participated": 6 - } - }, - "completed_tournaments": [ - { - "tournament_number": 1, - "finished_at": "2025-08-01T10:09:50.293741", - "results_summary": { - "participants": 3, - "total_shots": 120 - } - }, - { - "tournament_number": 2, - "finished_at": "2025-08-01T10:10:01.727450", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 3, - "finished_at": "2025-08-01T10:10:14.615370", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 4, - "finished_at": "2025-08-01T10:10:24.445436", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 5, - "finished_at": "2025-08-01T10:10:34.780992", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - }, - { - "tournament_number": 6, - "finished_at": "2025-08-01T10:10:45.109247", - "results_summary": { - "participants": 4, - "total_shots": 160 - } - } - ], - "league_finished": true, - "finished_at": "2025-08-01T10:10:45.109249" - }, - "archived_at": "2025-08-01T10:10:45.109575" -} \ No newline at end of file diff --git a/league_archives/league_20250802_152240.json b/league_archives/league_20250802_152240.json deleted file mode 100644 index 8221333..0000000 --- a/league_archives/league_20250802_152240.json +++ /dev/null @@ -1,86 +0,0 @@ -{ - "league": { - "league_id": "league_20250802_151950", - "created_at": "2025-08-02T15:19:50.435781", - "tournament_type": "20_targets", - "total_tournaments": 6, - "current_tournament": 1, - "participants": { - "1": { - "name": "Domen Pleterski", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "2": { - "name": "Nik Pleterski", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "3": { - "name": "Ivan Tandler", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "4": { - "name": "Mateja Pleterski", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "5": { - "name": "Jo\u017ee Verhnjak", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "6": { - "name": "Mateja Senica", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "7": { - "name": "Branko Poker\u017enik", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "8": { - "name": "Franc \u017digart", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - }, - "9": { - "name": "Janez Bo\u017ei\u010d", - "joker_used": false, - "tournament_results": [], - "total_score": 0, - "final_score": 0, - "tournaments_participated": 0 - } - }, - "completed_tournaments": [], - "league_finished": false - }, - "archived_at": "2025-08-02T15:22:40.774320" -} \ No newline at end of file diff --git a/league_archives/league_20250920_153253.json b/league_archives/league_20250920_153253.json new file mode 100644 index 0000000..113bc04 --- /dev/null +++ b/league_archives/league_20250920_153253.json @@ -0,0 +1,1975 @@ +{ + "league": { + "league_id": "league_20250920_153104", + "created_at": "2025-09-20T15:31:04.050964", + "tournament_type": "4_targets", + "total_tournaments": 5, + "current_tournament": 5, + "participants": { + "1": { + "name": "Domen Pleterski", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 2, + "score": 86, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 126, + "tens_count": 4, + "participated": true + }, + { + "tournament": 4, + "score": 126, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 95, + "tens_count": 1, + "participated": true + } + ], + "total_score": 433, + "final_score": 433, + "tournaments_participated": 4 + }, + "2": { + "name": "Nik Pleterski", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 2, + "score": 75, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 118, + "tens_count": 2, + "participated": true + }, + { + "tournament": 4, + "score": 109, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 77, + "tens_count": 1, + "participated": true + } + ], + "total_score": 379, + "final_score": 379, + "tournaments_participated": 4 + }, + "3": { + "name": "Ivan Tandler", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 2, + "score": 119, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 104, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 70, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 112, + "tens_count": 4, + "participated": true + } + ], + "total_score": 405, + "final_score": 405, + "tournaments_participated": 4 + }, + "4": { + "name": "Mateja Pleterski", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 2, + "score": 95, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 102, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 105, + "tens_count": 2, + "participated": true + }, + { + "tournament": 5, + "score": 72, + "tens_count": 2, + "participated": true + } + ], + "total_score": 374, + "final_score": 374, + "tournaments_participated": 4 + }, + "5": { + "name": "Jo\u017ee Verhnjak", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 102, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 82, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 104, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 91, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 101, + "tens_count": 2, + "participated": true + } + ], + "total_score": 480, + "final_score": 398, + "tournaments_participated": 5 + }, + "6": { + "name": "Mateja Senica", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 91, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 100, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 133, + "tens_count": 4, + "participated": true + }, + { + "tournament": 4, + "score": 110, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 95, + "tens_count": 3, + "participated": true + } + ], + "total_score": 529, + "final_score": 438, + "tournaments_participated": 5 + }, + "7": { + "name": "Branko Poker\u017enik", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 69, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 100, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 104, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 95, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 110, + "tens_count": 1, + "participated": true + } + ], + "total_score": 478, + "final_score": 409, + "tournaments_participated": 5 + }, + "8": { + "name": "Franc \u017digart", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 106, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 101, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 86, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 111, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 105, + "tens_count": 1, + "participated": true + } + ], + "total_score": 509, + "final_score": 423, + "tournaments_participated": 5 + }, + "9": { + "name": "Janez Bo\u017ei\u010d", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 122, + "tens_count": 6, + "participated": true + }, + { + "tournament": 2, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 3, + "score": 91, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 95, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 102, + "tens_count": 3, + "participated": true + } + ], + "total_score": 410, + "final_score": 410, + "tournaments_participated": 4 + }, + "10": { + "name": "Mitja \u010ceh", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 108, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 3, + "score": 99, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 90, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 90, + "tens_count": 0, + "participated": true + } + ], + "total_score": 387, + "final_score": 387, + "tournaments_participated": 4 + }, + "11": { + "name": "Rado Kefer", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 76, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 3, + "score": 98, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 91, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 97, + "tens_count": 3, + "participated": true + } + ], + "total_score": 362, + "final_score": 362, + "tournaments_participated": 4 + }, + "12": { + "name": "Matej Kvasnik", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 108, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 3, + "score": 95, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 131, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 91, + "tens_count": 0, + "participated": true + } + ], + "total_score": 425, + "final_score": 425, + "tournaments_participated": 4 + }, + "13": { + "name": "Angelca Mrak", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 115, + "tens_count": 5, + "participated": true + }, + { + "tournament": 2, + "score": 89, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 114, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 75, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 393, + "final_score": 393, + "tournaments_participated": 4 + }, + "14": { + "name": "Karli Proje", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 107, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 107, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 105, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 119, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 438, + "final_score": 438, + "tournaments_participated": 4 + }, + "15": { + "name": "Jan Pleterski", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 117, + "tens_count": 4, + "participated": true + }, + { + "tournament": 2, + "score": 77, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 73, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 98, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 365, + "final_score": 365, + "tournaments_participated": 4 + }, + "16": { + "name": "Silvo Poro\u010dnik", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 107, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 94, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 91, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 107, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 399, + "final_score": 399, + "tournaments_participated": 4 + }, + "17": { + "name": "Du\u0161an Onuk", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 86, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 93, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 102, + "tens_count": 2, + "participated": true + }, + { + "tournament": 4, + "score": 90, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 371, + "final_score": 371, + "tournaments_participated": 4 + }, + "18": { + "name": "Matja\u017e Pleterski", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 115, + "tens_count": 4, + "participated": true + }, + { + "tournament": 2, + "score": 92, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 82, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 112, + "tens_count": 2, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 401, + "final_score": 401, + "tournaments_participated": 4 + }, + "19": { + "name": "Franc Rizmal", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 93, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 110, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 94, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 114, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 411, + "final_score": 411, + "tournaments_participated": 4 + }, + "20": { + "name": "Jo\u017ee Preglav", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 134, + "tens_count": 3, + "participated": true + }, + { + "tournament": 2, + "score": 95, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 90, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 122, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + } + ], + "total_score": 441, + "final_score": 441, + "tournaments_participated": 4 + }, + "21": { + "name": "Marko Blimen", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 109, + "tens_count": 3, + "participated": true + }, + { + "tournament": 2, + "score": 103, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 104, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 121, + "tens_count": 5, + "participated": true + } + ], + "total_score": 437, + "final_score": 437, + "tournaments_participated": 4 + }, + "22": { + "name": "Doris Fesel", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 94, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 107, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 95, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 104, + "tens_count": 1, + "participated": true + } + ], + "total_score": 400, + "final_score": 400, + "tournaments_participated": 4 + }, + "23": { + "name": "Robi Krautberger", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 100, + "tens_count": 4, + "participated": true + }, + { + "tournament": 2, + "score": 118, + "tens_count": 5, + "participated": true + }, + { + "tournament": 3, + "score": 107, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 85, + "tens_count": 2, + "participated": true + } + ], + "total_score": 410, + "final_score": 410, + "tournaments_participated": 4 + }, + "24": { + "name": "Jo\u017ee Verdinek", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 103, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 89, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 92, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 104, + "tens_count": 1, + "participated": true + } + ], + "total_score": 388, + "final_score": 388, + "tournaments_participated": 4 + }, + "25": { + "name": "Andrej Herman", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 95, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 90, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 85, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 110, + "tens_count": 2, + "participated": true + } + ], + "total_score": 380, + "final_score": 380, + "tournaments_participated": 4 + }, + "26": { + "name": "Jakob Herman", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 102, + "tens_count": 3, + "participated": true + }, + { + "tournament": 2, + "score": 107, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 91, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 102, + "tens_count": 1, + "participated": true + } + ], + "total_score": 402, + "final_score": 402, + "tournaments_participated": 4 + }, + "27": { + "name": "Janez Mrak", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 95, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 105, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 106, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 93, + "tens_count": 1, + "participated": true + } + ], + "total_score": 399, + "final_score": 399, + "tournaments_participated": 4 + }, + "28": { + "name": "An\u017ee Kolar", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 122, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 112, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 119, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 5, + "score": 124, + "tens_count": 3, + "participated": true + } + ], + "total_score": 477, + "final_score": 477, + "tournaments_participated": 4 + }, + "30": { + "name": "Maja Hirtl", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 90, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 80, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 90, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 114, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 100, + "tens_count": 2, + "participated": true + } + ], + "total_score": 474, + "final_score": 394, + "tournaments_participated": 5 + }, + "31": { + "name": "Dejan Ku\u010dnik", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 81, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 106, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 113, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 86, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 92, + "tens_count": 1, + "participated": true + } + ], + "total_score": 478, + "final_score": 397, + "tournaments_participated": 5 + }, + "32": { + "name": "David Strni\u0161a", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 108, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 108, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 130, + "tens_count": 4, + "participated": true + }, + { + "tournament": 4, + "score": 116, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 104, + "tens_count": 1, + "participated": true + } + ], + "total_score": 566, + "final_score": 462, + "tournaments_participated": 5 + }, + "33": { + "name": "Namir Uzunovi\u0107", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 116, + "tens_count": 3, + "participated": true + }, + { + "tournament": 2, + "score": 100, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 74, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 93, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 93, + "tens_count": 2, + "participated": true + } + ], + "total_score": 476, + "final_score": 402, + "tournaments_participated": 5 + }, + "34": { + "name": "Jo\u017ee Planin\u0161ec", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 110, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 82, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 87, + "tens_count": 2, + "participated": true + }, + { + "tournament": 4, + "score": 120, + "tens_count": 5, + "participated": true + }, + { + "tournament": 5, + "score": 105, + "tens_count": 5, + "participated": true + } + ], + "total_score": 504, + "final_score": 422, + "tournaments_participated": 5 + }, + "35": { + "name": "Vanja Kolar", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 72, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 97, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 104, + "tens_count": 2, + "participated": true + }, + { + "tournament": 4, + "score": 117, + "tens_count": 2, + "participated": true + }, + { + "tournament": 5, + "score": 115, + "tens_count": 2, + "participated": true + } + ], + "total_score": 505, + "final_score": 433, + "tournaments_participated": 5 + }, + "36": { + "name": "Klara Wankmuller", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 109, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 115, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 93, + "tens_count": 1, + "participated": true + }, + { + "tournament": 4, + "score": 107, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 99, + "tens_count": 2, + "participated": true + } + ], + "total_score": 523, + "final_score": 430, + "tournaments_participated": 5 + }, + "37": { + "name": "Milan Stramec", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 120, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 75, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 98, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 75, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 111, + "tens_count": 2, + "participated": true + } + ], + "total_score": 479, + "final_score": 404, + "tournaments_participated": 5 + }, + "38": { + "name": "Bojan Sudar", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 108, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 84, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 116, + "tens_count": 4, + "participated": true + }, + { + "tournament": 5, + "score": 84, + "tens_count": 2, + "participated": true + } + ], + "total_score": 392, + "final_score": 392, + "tournaments_participated": 4 + }, + "39": { + "name": "Tia Sudar", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 110, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 72, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 86, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 96, + "tens_count": 1, + "participated": true + } + ], + "total_score": 364, + "final_score": 364, + "tournaments_participated": 4 + }, + "40": { + "name": "Jaka Cvar", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 99, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 92, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 112, + "tens_count": 5, + "participated": true + }, + { + "tournament": 5, + "score": 92, + "tens_count": 0, + "participated": true + } + ], + "total_score": 395, + "final_score": 395, + "tournaments_participated": 4 + }, + "41": { + "name": "Tadej \u0160truc", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 78, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 97, + "tens_count": 3, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 114, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 101, + "tens_count": 0, + "participated": true + } + ], + "total_score": 390, + "final_score": 390, + "tournaments_participated": 4 + }, + "42": { + "name": "Jure Glaser", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 109, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 66, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 86, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 138, + "tens_count": 5, + "participated": true + } + ], + "total_score": 399, + "final_score": 399, + "tournaments_participated": 4 + }, + "43": { + "name": "Marko Pokr\u017enik", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 104, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 104, + "tens_count": 2, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 85, + "tens_count": 2, + "participated": true + }, + { + "tournament": 5, + "score": 106, + "tens_count": 1, + "participated": true + } + ], + "total_score": 399, + "final_score": 399, + "tournaments_participated": 4 + }, + "44": { + "name": "Anka Ka\u010dnik", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 92, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 83, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 95, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 111, + "tens_count": 2, + "participated": true + } + ], + "total_score": 381, + "final_score": 381, + "tournaments_participated": 4 + }, + "45": { + "name": "Lidija Blimen", + "joker_used": true, + "tournament_results": [ + { + "tournament": 1, + "score": 83, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 107, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 0, + "tens_count": 0, + "participated": false, + "joker": true + }, + { + "tournament": 4, + "score": 104, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 92, + "tens_count": 3, + "participated": true + } + ], + "total_score": 386, + "final_score": 386, + "tournaments_participated": 4 + }, + "46": { + "name": "Tijana \u0160tumpfl", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 91, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 90, + "tens_count": 0, + "participated": true + }, + { + "tournament": 3, + "score": 102, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 100, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 140, + "tens_count": 5, + "participated": true + } + ], + "total_score": 523, + "final_score": 433, + "tournaments_participated": 5 + }, + "47": { + "name": "Ljuba Mr\u0161ak", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 82, + "tens_count": 1, + "participated": true + }, + { + "tournament": 2, + "score": 96, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 86, + "tens_count": 0, + "participated": true + }, + { + "tournament": 4, + "score": 76, + "tens_count": 0, + "participated": true + }, + { + "tournament": 5, + "score": 94, + "tens_count": 4, + "participated": true + } + ], + "total_score": 434, + "final_score": 358, + "tournaments_participated": 5 + }, + "48": { + "name": "Janja Salcman", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 101, + "tens_count": 2, + "participated": true + }, + { + "tournament": 2, + "score": 99, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 105, + "tens_count": 2, + "participated": true + }, + { + "tournament": 4, + "score": 96, + "tens_count": 1, + "participated": true + }, + { + "tournament": 5, + "score": 93, + "tens_count": 1, + "participated": true + } + ], + "total_score": 494, + "final_score": 401, + "tournaments_participated": 5 + }, + "49": { + "name": "Jolanda Verhnjak", + "joker_used": false, + "tournament_results": [ + { + "tournament": 1, + "score": 95, + "tens_count": 0, + "participated": true + }, + { + "tournament": 2, + "score": 93, + "tens_count": 1, + "participated": true + }, + { + "tournament": 3, + "score": 112, + "tens_count": 3, + "participated": true + }, + { + "tournament": 4, + "score": 107, + "tens_count": 3, + "participated": true + }, + { + "tournament": 5, + "score": 109, + "tens_count": 1, + "participated": true + } + ], + "total_score": 516, + "final_score": 423, + "tournaments_participated": 5 + } + }, + "completed_tournaments": [ + { + "tournament_number": 1, + "tournament_type": "4_targets", + "finished_at": "2025-09-20T15:31:25.272296", + "results_summary": { + "participants": 44, + "shots_per_participant": 20, + "total_shots": 880, + "format_description": "4 Targets (5 shots each)" + } + }, + { + "tournament_number": 2, + "tournament_type": "4_targets", + "finished_at": "2025-09-20T15:31:46.364981", + "results_summary": { + "participants": 44, + "shots_per_participant": 20, + "total_shots": 880, + "format_description": "4 Targets (5 shots each)" + } + }, + { + "tournament_number": 3, + "tournament_type": "4_targets", + "finished_at": "2025-09-20T15:32:08.801733", + "results_summary": { + "participants": 40, + "shots_per_participant": 20, + "total_shots": 800, + "format_description": "4 Targets (5 shots each)" + } + }, + { + "tournament_number": 4, + "tournament_type": "4_targets", + "finished_at": "2025-09-20T15:32:30.303850", + "results_summary": { + "participants": 40, + "shots_per_participant": 20, + "total_shots": 800, + "format_description": "4 Targets (5 shots each)" + } + }, + { + "tournament_number": 5, + "tournament_type": "4_targets", + "finished_at": "2025-09-20T15:32:53.267295", + "results_summary": { + "participants": 40, + "shots_per_participant": 20, + "total_shots": 800, + "format_description": "4 Targets (5 shots each)" + } + } + ], + "league_finished": true, + "finished_at": "2025-09-20T15:32:53.267298" + }, + "archived_at": "2025-09-20T15:32:53.268502" +} \ No newline at end of file diff --git a/locales/en.json b/locales/en.json new file mode 100644 index 0000000..dd1a3c0 --- /dev/null +++ b/locales/en.json @@ -0,0 +1,362 @@ +{ + "general": { + "loading": "Loading...", + "save": "Save", + "cancel": "Cancel", + "close": "Close", + "back": "Back", + "next": "Next", + "previous": "Previous", + "confirm": "Confirm", + "delete": "Delete", + "edit": "Edit", + "add": "Add", + "remove": "Remove", + "settings": "Settings", + "yes": "Yes", + "no": "No", + "all": "All", + "status": "Status", + "actions": "Actions", + "select_all": "Select All", + "enable_selected": "Enable Selected", + "disable_selected": "Disable Selected", + "print": "Print", + "visible": "Visible", + "date": "Date", + "view": "View" + }, + "navigation": { + "dashboard": "Dashboard", + "tournament": "Tournament", + "league": "League", + "results": "Results", + "players": "Players", + "analysis": "Analysis", + "archive": "Archive", + "calculator": "Results Calculator", + "draft": "Draft", + "streams": "Streams" + }, + "camera": { + "title": "Camera Dashboard", + "camera": "Camera", + "stream": "Stream", + "fullscreen": "Fullscreen", + "settings": "Camera Settings", + "titles": "Camera Titles", + "show_titles": "Show Titles", + "title_size": "Title Size", + "unavailable": "Unavailable", + "show_card_titles": "Show Card Titles", + "title_text_size": "Title Text Size", + "language_settings": "Language Settings", + "display_options": "Display Options", + "click_to_view_fullscreen": "Click to view fullscreen" + }, + "tournament": { + "tournament": "Tournament", + "round": "Round", + "round_of": "of", + "previous_round": "Previous Round", + "next_round": "Next Round", + "current_round": "Current Round", + "total_rounds": "Total Rounds", + "active_tournament": "Active Tournament", + "tournament_management": "Tournament Management", + "tournament_type": "Tournament Type", + "participants": "Participants", + "start_tournament": "Start Tournament", + "reset_tournament": "Reset Tournament", + "tournament_setup": "Tournament Setup", + "view_draft": "Draft", + "manage_tournament": "Manage Tournament", + "tournament_results": "Tournament Results", + "tournament_finished": "Tournament Finished", + "tournament_scores": "Tournament Scores", + "view_full_tournament_draft": "Draft", + "tournament_mode": "Tournament Mode", + "active_tournament": "Active Tournament", + "current_round_info": "Camera cards show current round players", + "manage_tournament": "Manage Tournament", + "reset_tournament": "Reset Tournament", + "current_tournament": "Current Tournament", + "completed": "Completed", + "created": "Created", + "tournaments": "Tournaments", + "league_tournament": "League Tournament", + "finished": "Finished" + }, + "tournament_types": { + "4_targets": "4 Targets", + "20_targets": "20 Targets", + "40_targets": "40 Targets", + "4_targets_desc": "Quick format with 4 targets, 5 shots each (20 shots total)", + "20_targets_desc": "Standard format with 20 targets, 2 shots each (40 shots total)", + "40_targets_desc": "Extended format with 40 targets, 2 shots each (80 shots total)", + "4_targets_full": "4 Targets (5 shots each)", + "20_targets_full": "20 Targets (2 shots each)", + "40_targets_full": "40 Targets (2 shots each)", + "40_target_tournaments": "40-Target Tournaments", + "20_target_tournaments": "20-Target Tournaments", + "4_target_tournaments": "4-Target Tournaments" + }, + "league": { + "league": "League", + "league_championship": "League Championship", + "league_management": "League Management", + "league_active": "League Active", + "league_finished": "League Finished", + "start_league": "Start League", + "reset_league": "Reset League", + "current_tournament": "Current Tournament", + "total_tournaments": "Total Tournaments", + "completed_tournaments": "Completed Tournaments", + "completed_leagues": "Completed Leagues", + "league_results": "League Results", + "final_rankings": "Final Rankings", + "final_rankings_best_4_of_5": "Final Rankings - Best 4 of 5 Tournaments", + "best_4_of_5": "Best 4 of 5 Tournaments", + "league_champions": "League Champions", + "participant": "Participant", + "tournament_scores": "Tournament Scores", + "final": "Final", + "best_4": "Best 4", + "total_10s": "Total 10s", + "scoring_legend": "Scoring Legend", + "counted_score": "Counted Score", + "joker_used": "Joker Used", + "final_score": "Final Score", + "highest_score": "Highest Score", + "average_final": "Average Final", + "5_tournament_league": "5 Tournament League - Best 4 Count", + "joker_used_badge": "Joker Used", + "search_players_placeholder": "Search players by name...", + "no_players_found": "No players found matching your search criteria.", + "no_players_selected": "No players selected", + "start_tournament_number": "Start Tournament", + "setup": "Setup", + "excluded_worst": "Excluded (Worst)", + "total_10s_tiebreaker": "Total 10s (Tiebreaker)", + "joker_selection_for_tournament": "Joker Selection for Tournament", + "joker_instructions": "Select players who will use their joker (skip this tournament). Each player can only use their joker once per league.", + "start_tournament_confirm_single": "Start next tournament? 1 player will use their joker.", + "start_tournament_confirm_multiple": "Start next tournament? {count} players will use their joker.", + "no_active_league_tournament": "No Active League or Tournament", + "select_tournament_type": "Select Tournament Type", + "add_new_player": "Add New Player", + "enter_player_name": "Enter player name...", + "start_league_5_tournaments": "Start League (5 Tournaments)", + "start_single_tournament": "Start Single Tournament", + "id": "ID", + "name": "Name", + "edit": "Edit", + "system": "System", + "camera": "Camera", + "tournaments": "Tournaments", + "results.most_tens": "Most 10s" + }, + "results": { + "results": "Results", + "final_results": "Final Results", + "rankings": "Rankings", + "score": "Score", + "total_score": "Total Score", + "shots": "Shots", + "total_shots": "Total Shots", + "tens": "Tens", + "most_tens": "Most Tens", + "best_score": "Best Score", + "average_score": "Average Score", + "highest_score": "Highest Score", + "worst_score": "Worst Score", + "completed": "Completed", + "position": "Position", + "points": "Points", + "top_3_winners": "Top 3 Winners" + }, + "players": { + "player": "Player", + "players": "Players", + "player_name": "Player Name", + "add_player": "Add Player", + "edit_player": "Edit Player", + "delete_player": "Delete Player", + "enabled": "Enabled", + "disabled": "Disabled", + "player_management": "Player Management", + "player_analysis": "Player Analysis", + "player_stats": "Player Stats", + "competing": "Competing", + "not_competing": "Not Competing", + "search_players": "Search Players", + "total_players": "Total Players", + "enabled_players": "Enabled Players", + "disabled_players": "Disabled Players", + "players_label": "Players" + }, + "analysis": { + "analysis": "Analysis", + "statistics": "Statistics", + "performance": "Performance", + "comparison": "Comparison", + "trends": "Trends", + "shot_accuracy": "Shot Accuracy", + "tournament_history": "Tournament History", + "overall_stats": "Overall Stats", + "tournament_leaders": "Tournament Leaders", + "overall_champions": "Overall Champions", + "by_tournament_type": "By Tournament Type", + "all_players": "All Players", + "select_player": "Select a Player to Analyze", + "unable_load_data": "Unable to Load Tournament Data", + "no_tournament_data": "No Tournament Data Available", + "best_score": "Best Score", + "most_tens": "Most 10s", + "overview_champions": "Overall Champions by Tournament Type", + "unknown_date": "Unknown Date", + "sort_by_name": "Sort by Name", + "sort_by_best_score": "Sort by Best Score", + "sort_by_average_score": "Sort by Average Score", + "sort_by_total_tournaments": "Sort by Total Tournaments", + "sort_by_total_leagues": "Sort by Total Leagues", + "sort_by_total_shots": "Sort by Total Shots", + "best_score_label": "Best Score:", + "total_tournaments": "Total Tournaments" + }, + "scoring": { + "target": "Target", + "shot": "Shot", + "value": "Value", + "total": "Total", + "calculate": "Calculate", + "calculator": "Calculator", + "results_calculator": "Results Calculator", + "enter_scores": "Enter Scores", + "finish_tournament": "Finish Tournament", + "tournament_scoring": "Tournament Scoring", + "total_shots": "Total Shots", + "fill_all_players_random": "Fill All Players Random", + "not_started": "Not Started", + "in_progress": "In Progress", + "clear_all": "Clear All", + "save_progress": "Save Progress", + "finish_tournament_section": "Finish Tournament", + "finish_tournament_show_results": "Finish Tournament & Show Results", + "saving_all": "Saving All...", + "all_saved": "All Saved!", + "filling_all": "Filling All...", + "all_filled": "All Filled!", + "finishing_tournament": "Finishing Tournament...", + "completed": "Completed", + "points": "Points", + "clear": "Clear", + "save": "Save", + "warning": "Warning", + "finish_warning": "Not all participants have completed scores. Please ensure all scoring is complete before finishing.", + "finish_tournament_button": "Finish Tournament & Show Results", + "enter_scores_40_targets": "Enter scores for each participant (40 targets, 2 shots each). Score 0 = miss.", + "enter_scores_20_targets": "Enter scores for each participant (20 targets, 2 shots each). Score 0 = miss.", + "enter_scores_4_targets": "Enter scores for each participant (4 targets, 5 shots each). Score 0 = miss." + }, + "mobile": { + "mobile_streams": "Mobile Streams", + "camera_streams": "Camera Streams", + "mobile_disabled": "Mobile Support Disabled", + "use_desktop": "Use Desktop Version", + "redirecting": "Redirecting...", + "rotate_landscape": "Rotate to landscape or tap for fullscreen" + }, + "messages": { + "success": "Success!", + "error": "Error!", + "warning": "Warning!", + "info": "Information", + "confirm_delete": "Are you sure you want to delete?", + "confirm_reset": "Are you sure you want to reset?", + "confirm_start_tournament": "Start single tournament with {players} players using {format} format?", + "confirm_start_league": "Start league with {players} players using {format} format?", + "tournament_started": "Tournament started successfully!", + "league_championships": "League Championships", + "other_tournaments": "Other Tournaments", + "no_archives_found": "No Archives Found", + "complete_tournaments_msg": "Complete some tournaments or leagues to see them archived here", + "start_tournament": "Start Tournament", + "archive_deleted": "Archive deleted successfully", + "confirm_delete_archive": "Are you sure you want to delete this {type}? This action cannot be undone.", + "unknown_date": "Unknown Date", + "league_championship": "League Championship", + "40_target_tournament": "40-Target Tournament", + "20_target_tournament": "20-Target Tournament", + "4_target_tournament": "4-Target Tournament", + "unknown": "Unknown", + "edit_archive": "Edit Archive", + "archive_name": "Archive Name", + "enter_archive_name": "Enter archive name", + "format_type": "Format Type", + "target_format": "Target Format", + "save_changes": "Save Changes", + "delete_archive": "Delete Archive", + "are_you_sure_delete": "Are you sure you want to delete this", + "action_cannot_undone": "This action cannot be undone", + "confirm_action": "Confirm Action", + "are_you_sure_proceed": "Are you sure you want to proceed?", + "single_elimination": "Single Elimination", + "double_elimination": "Double Elimination", + "round_robin": "Round Robin", + "swiss_system": "Swiss System", + "tournament_finished": "Tournament finished successfully!", + "league_started": "League started successfully!", + "league_finished": "League finished successfully!", + "settings_saved": "Settings saved!", + "player_added": "Player added!", + "player_updated": "Player updated!", + "player_deleted": "Player deleted!", + "no_data": "No data", + "loading_data": "Loading data...", + "connection_error": "Connection error", + "refresh_required": "Page refresh required" + }, + "draft": { + "tournament_draft": "Tournament Draft", + "shooting_tournament": "Shooting Tournament", + "current": "Current", + "done": "Done", + "wait": "Wait", + "round": "Round", + "of": "of", + "previous": "Previous", + "next": "Next", + "players": "players", + "rounds": "rounds", + "currently_on_round": "Currently on Round", + "empty": "Empty", + "no_active_tournament": "No Active Tournament", + "setup_tournament_message": "Go to Tournament Management to set up players and start a tournament.", + "set_up_tournament": "Set Up Tournament", + "updating": "Updating...", + "manage": "Manage", + "dashboard": "Dashboard" + }, + "time": { + "monday": "Monday", + "tuesday": "Tuesday", + "wednesday": "Wednesday", + "thursday": "Thursday", + "friday": "Friday", + "saturday": "Saturday", + "sunday": "Sunday", + "january": "January", + "february": "February", + "march": "March", + "april": "April", + "may": "May", + "june": "June", + "july": "July", + "august": "August", + "september": "September", + "october": "October", + "november": "November", + "december": "December" + } +} \ No newline at end of file diff --git a/locales/sl.json b/locales/sl.json new file mode 100644 index 0000000..3d6d67e --- /dev/null +++ b/locales/sl.json @@ -0,0 +1,369 @@ +{ + "general": { + "loading": "Nalaganje...", + "save": "Shrani", + "cancel": "Prekliči", + "close": "Zapri", + "back": "Nazaj", + "next": "Naprej", + "previous": "Prejšnji", + "confirm": "Potrdi", + "delete": "Izbriši", + "edit": "Uredi", + "add": "Dodaj", + "remove": "Odstrani", + "settings": "Nastavitve", + "yes": "Da", + "no": "Ne", + "all": "Vse", + "status": "Status", + "actions": "Dejanja", + "select_all": "Izberi Vse", + "enable_selected": "Omogoči Izbrane", + "disable_selected": "Onemogoči Izbrane", + "print": "Natisni", + "visible": "Vidni", + "date": "Datum", + "view": "Oglej" + }, + "navigation": { + "dashboard": "Nadzorna Plošča", + "tournament": "Turnir", + "league": "Liga", + "results": "Rezultati", + "players": "Igralci", + "analysis": "Analiza", + "archive": "Arhiv", + "calculator": "Kalkulator Rezultatov", + "draft": "Žreb", + "streams": "Prenosi" + }, + "camera": { + "title": "", + "camera": "Kamera", + "stream": "Prenos", + "fullscreen": "Celozaslonski Način", + "settings": "Nastavitve Kamer", + "titles": "Naslovi Kamer", + "show_titles": "Prikaži Naslove", + "title_size": "Velikost Naslova", + "unavailable": "Nedostopno", + "show_card_titles": "Prikaži Naslove Kartic", + "title_text_size": "Velikost Besedila Naslova", + "language_settings": "Jezikovne Nastavitve", + "display_options": "Možnosti Prikaza", + "click_to_view_fullscreen": "Klikni za celozaslonski prikaz" + }, + "tournament": { + "tournament": "Turnir", + "round": "Krog", + "round_of": "od", + "previous_round": "Prejšnji Krog", + "next_round": "Naslednji Krog", + "current_round": "Trenutni Krog", + "total_rounds": "Skupaj Krogov", + "active_tournament": "Aktiven Turnir", + "tournament_management": "Upravljanje Turnirja", + "tournament_type": "Tip Turnirja", + "participants": "Udeleženci", + "start_tournament": "Začni Turnir", + "reset_tournament": "Ponastavi Turnir", + "tournament_setup": "Nastavitev Turnirja", + "view_draft": "Žreb", + "manage_tournament": "Upravljaj Turnir", + "tournament_results": "Rezultati Turnirja", + "tournament_finished": "Turnir Zaključen", + "tournament_scores": "Rezultati Turnirja", + "view_full_tournament_draft": "Žreb", + "tournament_mode": "Način Turnirja", + "current_round_info": "Kartice kamer prikazujejo igralce trenutnega kroga", + "current_tournament": "Trenutni Turnir", + "completed": "Zaključeno", + "created": "Ustvarjeno", + "tournaments": "Turnirji", + "league_tournament": "Ligaški Turnir", + "start_league_5": "Začni Ligo (5 Turnirjev)", + "start_single": "Začni Posamezen Turnir", + "league_completed": "Liga Zaključena", + "section_title": "Sekcija", + "finished": "Zaključeno", + "targets": "Število Tarč" + }, + "tournament_types": { + "4_targets": "4 Tarče", + "20_targets": "20 Tarč", + "40_targets": "40 Tarč", + "4_targets_desc": "Hitri format s 4 tarčami, 5 strelov na tarčo (20 strelov skupaj)", + "20_targets_desc": "Standardni format z 20 tarčami, 2 strela na tarčo (40 strelov skupaj)", + "40_targets_desc": "Razširjeni format s 40 tarčami, 2 strela na tarčo (80 strelov skupaj)", + "4_targets_full": "4 Tarče (5 strelov vsaka)", + "20_targets_full": "20 Tarč (2 strela vsaka)", + "40_targets_full": "40 Tarč (2 strela vsaka)", + "40_target_tournaments": "40-Tarčni Turnirji", + "20_target_tournaments": "20-Tarčni Turnirji", + "4_target_tournaments": "4-Tarčni Turnirji" + }, + "league": { + "league": "Liga", + "league_championship": "Liga", + "league_management": "Upravljanje Lige", + "league_active": "Liga Aktivna", + "league_finished": "Liga Zaključena", + "start_league": "Začni Ligo", + "reset_league": "Ponastavi Ligo", + "current_tournament": "Trenutni Turnir", + "total_tournaments": "Skupaj Turnirjev", + "completed_tournaments": "Zaključeni Turnirji", + "completed_leagues": "Zaključene Lige", + "league_results": "Rezultati Lige", + "final_rankings": "Končne Uvrstitve", + "final_rankings_best_4_of_5": "Končne Uvrstitve - Najboljši 4 od 5 Turnirjev", + "best_4_of_5": "Najboljši 4 od 5 Turnirjev", + "league_complete": "Liga Zaključena!", + "league_complete_info": "Vsi turnirji načrtovani. Zaključi trenutnega za končne rezultate.", + "view_league_results": "Oglej si Rezultate Lige", + "score_tournament": "Točkuj Turnir", + "league_champions": "Ligaški Prvaki", + "participant": "Udeleženec", + "tournament_scores": "Rezultati Turnirjev", + "final": "Končni", + "best_4": "Najboljši 4", + "total_10s": "Desetke", + "scoring_legend": "Legenda Točkovanja", + "counted_score": "Rezultat", + "joker_used": "Joker Uporabljen", + "final_score": "Končni Rezultat", + "highest_score": "Najvišji Rezultat", + "average_final": "Povprečni Končni", + "5_tournament_league": "5 Turnirska Liga - Štejejo Najboljši 4", + "joker_used_badge": "Joker Uporabljen", + "search_players_placeholder": "Išči igralce po imenu...", + "no_players_found": "Ni najdenih igralcev, ki bi ustrezali kriterijem iskanja.", + "no_players_selected": "Ni izbranih igralcev", + "start_tournament_number": "Začni Turnir", + "setup": "Nastavitev", + "excluded_worst": "Izključeno (Najslabši)", + "total_10s_tiebreaker": "Število 10k (Odločilno)", + "joker_selection_for_tournament": "Izbira Jokerja za Turnir", + "joker_instructions": "Izberite igralce, ki bodo uporabili svojega Jokerja (preskočili ta turnir). Vsak igralec lahko uporabi svojega Jokerja samo enkrat na ligo.", + "start_tournament_confirm_single": "Začni naslednji turnir? 1 igralec bo uporabil svojega Jokerja.", + "start_tournament_confirm_multiple": "Začni naslednji turnir? {count} igralcev bo uporabilo svojega Jokerja.", + "no_active_league_tournament": "Ni Aktivne Lige ali Turnirja", + "select_tournament_type": "Izberi Tip Turnirja", + "add_new_player": "Dodaj Novega Igralca", + "enter_player_name": "Vnesite ime igralca...", + "start_league_5_tournaments": "Začni Ligo (5 Turnirjev)", + "start_single_tournament": "Začni Posamezen Turnir", + "id": "ID", + "name": "Ime", + "edit": "Uredi", + "system": "Sistem", + "camera": "Kamera", + "tournaments": "Turnirji", + "results.most_tens": "Največ Desetk" + + }, + "results": { + "results": "Rezultati", + "final_results": "Končni Rezultati", + "rankings": "Uvrstitve", + "score": "Rezultat", + "total_score": "Skupni Rezultat", + "shots": "Streli", + "total_shots": "Skupaj Strelov", + "tens": "Desetke", + "most_tens": "Največ Desetk", + "best_score": "Najboljši Rezultat", + "average_score": "Povprečni Rezultat", + "highest_score": "Najvišji Rezultat", + "worst_score": "Najslabši Rezultat", + "completed": "Zaključeno", + "position": "Uvrstitev", + "points": "Točke", + "top_3_winners": "Top 3 Zmagovalci" + }, + "players": { + "player": "Igralec", + "players": "Igralci", + "player_name": "Ime Igralca", + "add_player": "Dodaj Igralca", + "edit_player": "Uredi Igralca", + "delete_player": "Izbriši Igralca", + "enabled": "Omogočen", + "disabled": "Onemogočen", + "player_management": "Upravljanje Igralcev", + "player_analysis": "Analiza Igralcev", + "player_stats": "Statistike Igralcev", + "competing": "Tekmuje", + "not_competing": "Ne tekmuje", + "search_players": "Išči Igralce", + "total_players": "Skupaj Igralcev", + "enabled_players": "Omogočeni Igralci", + "disabled_players": "Onemogočeni Igralci", + "players_label": "Igralci" + }, + "analysis": { + "analysis": "Analiza", + "statistics": "Statistike", + "performance": "Uspešnost", + "comparison": "Primerjava", + "trends": "Trendi", + "shot_accuracy": "Natančnost Strelov", + "tournament_history": "Zgodovina Turnirjev", + "overall_stats": "Splošne Statistike", + "tournament_leaders": "Vodilni v Turnirjih", + "overall_champions": "Splošni Prvaki", + "by_tournament_type": "Po Tipu Turnirja", + "all_players": "Vsi Igralci", + "select_player": "Izberi igralca za analizo", + "unable_load_data": "Ne morem naložiti podatkov turnirja", + "no_tournament_data": "Ni podatkov o turnirju", + "best_score": "Najboljši Rezultat", + "most_tens": "Največ Desetk", + "overview_champions": "Splošni prvaki po tipu turnirja", + "unknown_date": "Neznan Datum", + "sort_by_name": "Razvrsti po Imenu", + "sort_by_best_score": "Razvrsti po Najboljšem Rezultatu", + "sort_by_average_score": "Razvrsti po Povprečnem Rezultatu", + "sort_by_total_tournaments": "Razvrsti po Skupnih Turnirjih", + "sort_by_total_leagues": "Razvrsti po Skupnih Ligah", + "sort_by_total_shots": "Razvrsti po Skupnih Strelih", + "best_score_label": "Najboljši Rezultat:", + "total_tournaments": "Skupaj Turnirjev" + }, + "scoring": { + "target": "Tarča", + "shot": "Strel", + "value": "Vrednost", + "total": "Skupaj", + "calculate": "Izračunaj", + "calculator": "Kalkulator", + "results_calculator": "Kalkulator Rezultatov", + "enter_scores": "Vnesi Rezultate", + "finish_tournament": "Zaključi Turnir", + "tournament_scoring": "Točkovanje Turnirja", + "total_shots": "Skupaj Strelov", + "fill_all_players_random": "Napolni Vse Igralce Naključno", + "not_started": "Ni Začeto", + "in_progress": "V Teku", + "clear_all": "Počisti Vse", + "save_progress": "Shrani Napredek", + "finish_tournament_section": "Zaključi Turnir", + "finish_tournament_show_results": "Zaključi Turnir & Prikaži Rezultate", + "saving_all": "Shranjujem Vse...", + "all_saved": "Vse Shranjeno!", + "filling_all": "Polnim Vse...", + "all_filled": "Vse Napolnjeno!", + "finishing_tournament": "Zaključujem Turnir...", + "completed": "Zaključeno", + "points": "Točke", + "clear": "Počisti", + "save": "Shrani", + "warning": "Opozorilo", + "finish_warning": "Niso vsi udeleženci zaključili s točkovanjem. Prosimo, da poskrbite, da je vse točkovanje zaključeno pred zaključkom.", + "finish_tournament_button": "Zaključi Turnir & Prikaži Rezultate", + "enter_scores_40_targets": "Vnesi rezultate za vsakega udeleženca (40 tarč, 2 strela na tarčo). Rezultat 0 = zgrešeno.", + "enter_scores_20_targets": "Vnesi rezultate za vsakega udeleženca (20 tarč, 2 strela na tarčo). Rezultat 0 = zgrešeno.", + "enter_scores_4_targets": "Vnesi rezultate za vsakega udeleženca (4 tarče, 5 strelov na tarčo). Rezultat 0 = zgrešeno." + }, + "mobile": { + "mobile_streams": "Mobilni Prenosi", + "camera_streams": "Prenosi Kamer", + "mobile_disabled": "Mobilna Podpora Onemogočena", + "use_desktop": "Uporabi Namizno Različico", + "redirecting": "Preusmerjam...", + "rotate_landscape": "Obrni v ležeči način ali dotakni se za celozaslonski način" + }, + "messages": { + "success": "Uspešno!", + "error": "Napaka!", + "warning": "Opozorilo!", + "info": "Informacija", + "confirm_delete": "Ali ste prepričani, da želite izbrisati?", + "confirm_reset": "Ali ste prepričani, da želite ponastaviti?", + "confirm_start_tournament": "Začni posamezen turnir z {players} igralci v {format} formatu?", + "confirm_start_league": "Začni ligo z {players} igralci v {format} formatu?", + "tournament_started": "Turnir uspešno začet!", + "league_championships": "Ligaška Prvenstva", + "other_tournaments": "Drugi Turnirji", + "no_archives_found": "Ni Najdenih Arhivov", + "complete_tournaments_msg": "Zaključi nekaj turnirjev ali lig, da jih vidiš arhivirane tukaj", + "start_tournament": "Začni Turnir", + "archive_deleted": "Arhiv uspešno izbrisan", + "confirm_delete_archive": "Ali ste prepričani, da želite izbrisati ta {type}? To dejanje ni mogoče razveljaviti.", + "unknown_date": "Neznan Datum", + "league_championship": "Liga", + "40_target_tournament": "40-Tarčni Turnir", + "20_target_tournament": "20-Tarčni Turnir", + "4_target_tournament": "4-Tarčni Turnir", + "unknown": "Neznano", + "edit_archive": "Uredi Arhiv", + "archive_name": "Ime Arhiva", + "enter_archive_name": "Vnesite ime arhiva", + "format_type": "Tip Formata", + "target_format": "Format Tarč", + "save_changes": "Shrani Spremembe", + "delete_archive": "Izbriši Arhiv", + "are_you_sure_delete": "Ali ste prepričani, da želite izbrisati ta", + "action_cannot_undone": "To dejanje ni mogoče razveljaviti", + "confirm_action": "Potrdi Dejanje", + "are_you_sure_proceed": "Ali ste prepričani, da želite nadaljevati?", + "single_elimination": "Ena Eliminacija", + "double_elimination": "Dvojna Eliminacija", + "round_robin": "Vsak z Vsakim", + "swiss_system": "Švicarski Sistem", + "tournament_finished": "Turnir uspešno zaključen!", + "league_started": "Liga uspešno začeta!", + "league_finished": "Liga uspešno zaključena!", + "settings_saved": "Nastavitve shranjene!", + "player_added": "Igralec dodan!", + "player_updated": "Igralec posodobljen!", + "player_deleted": "Igralec izbrisan!", + "no_data": "Ni podatkov", + "loading_data": "Nalagam podatke...", + "connection_error": "Napaka povezave", + "refresh_required": "Potrebno je osvežiti stran" + }, + "draft": { + "tournament_draft": "Žreb Turnirja", + "shooting_tournament": "Strelski Turnir", + "current": "Trenutni", + "done": "Končano", + "wait": "Čaka", + "round": "Krog", + "of": "od", + "previous": "Prejšnji", + "next": "Naslednji", + "players": "igralcev", + "rounds": "krogov", + "currently_on_round": "Trenutno v krogu", + "empty": "Prazno", + "no_active_tournament": "Ni Aktivnega Turnirja", + "setup_tournament_message": "Pojdi v Upravljanje Turnirja za nastavitev igralcev in začetek turnirja.", + "set_up_tournament": "Nastavitev Turnirja", + "updating": "Posodabljam...", + "manage": "Upravljaj", + "dashboard": "Nadzorna Plošča" + }, + "time": { + "monday": "Ponedeljek", + "tuesday": "Torek", + "wednesday": "Sreda", + "thursday": "Četrtek", + "friday": "Petek", + "saturday": "Sobota", + "sunday": "Nedelja", + "january": "Januar", + "february": "Februar", + "march": "Marec", + "april": "April", + "may": "Maj", + "june": "Junij", + "july": "Julij", + "august": "Avgust", + "september": "September", + "october": "Oktober", + "november": "November", + "december": "December" + } +} \ No newline at end of file diff --git a/players.json b/players.json index 43ad48d..a2e8e3f 100644 --- a/players.json +++ b/players.json @@ -48,97 +48,97 @@ { "id": 10, "name": "Mitja \u010ceh", - "enabled": false + "enabled": true }, { "id": 11, "name": "Rado Kefer", - "enabled": false + "enabled": true }, { "id": 12, "name": "Matej Kvasnik", - "enabled": false + "enabled": true }, { "id": 13, "name": "Angelca Mrak", - "enabled": false + "enabled": true }, { "id": 14, "name": "Karli Proje", - "enabled": false + "enabled": true }, { "id": 15, "name": "Jan Pleterski", - "enabled": false + "enabled": true }, { "id": 16, "name": "Silvo Poro\u010dnik", - "enabled": false + "enabled": true }, { "id": 17, "name": "Du\u0161an Onuk", - "enabled": false + "enabled": true }, { "id": 18, "name": "Matja\u017e Pleterski", - "enabled": false + "enabled": true }, { "id": 19, "name": "Franc Rizmal", - "enabled": false + "enabled": true }, { "id": 20, "name": "Jo\u017ee Preglav", - "enabled": false + "enabled": true }, { "id": 21, "name": "Marko Blimen", - "enabled": false + "enabled": true }, { "id": 22, "name": "Doris Fesel", - "enabled": false + "enabled": true }, { "id": 23, "name": "Robi Krautberger", - "enabled": false + "enabled": true }, { "id": 24, "name": "Jo\u017ee Verdinek", - "enabled": false + "enabled": true }, { "id": 25, "name": "Andrej Herman", - "enabled": false + "enabled": true }, { "id": 26, "name": "Jakob Herman", - "enabled": false + "enabled": true }, { "id": 27, "name": "Janez Mrak", - "enabled": false + "enabled": true }, { "id": 28, "name": "An\u017ee Kolar", - "enabled": false + "enabled": true }, { "id": 29, @@ -148,72 +148,102 @@ { "id": 30, "name": "Maja Hirtl", - "enabled": false + "enabled": true }, { "id": 31, "name": "Dejan Ku\u010dnik", - "enabled": false + "enabled": true }, { "id": 32, "name": "David Strni\u0161a", - "enabled": false + "enabled": true }, { "id": 33, "name": "Namir Uzunovi\u0107", - "enabled": false + "enabled": true }, { "id": 34, "name": "Jo\u017ee Planin\u0161ec", - "enabled": false + "enabled": true }, { "id": 35, "name": "Vanja Kolar", - "enabled": false + "enabled": true }, { "id": 36, "name": "Klara Wankmuller", - "enabled": false + "enabled": true }, { "id": 37, "name": "Milan Stramec", - "enabled": false + "enabled": true }, { "id": 38, "name": "Bojan Sudar", - "enabled": false + "enabled": true }, { "id": 39, "name": "Tia Sudar", - "enabled": false + "enabled": true }, { "id": 40, "name": "Jaka Cvar", - "enabled": false + "enabled": true }, { "id": 41, "name": "Tadej \u0160truc", - "enabled": false + "enabled": true }, { "id": 42, "name": "Jure Glaser", - "enabled": false + "enabled": true }, { "id": 43, "name": "Marko Pokr\u017enik", - "enabled": false + "enabled": true + }, + { + "id": 44, + "name": "Anka Ka\u010dnik", + "enabled": true + }, + { + "id": 45, + "name": "Lidija Blimen", + "enabled": true + }, + { + "id": 46, + "name": "Tijana \u0160tumpfl", + "enabled": true + }, + { + "id": 47, + "name": "Ljuba Mr\u0161ak", + "enabled": true + }, + { + "id": 48, + "name": "Janja Salcman", + "enabled": true + }, + { + "id": 49, + "name": "Jolanda Verhnjak", + "enabled": true } ] } \ No newline at end of file diff --git a/settings.json b/settings.json new file mode 100644 index 0000000..401b31e --- /dev/null +++ b/settings.json @@ -0,0 +1,3 @@ +{ + "header_text_size": "14px" +} \ No newline at end of file diff --git a/static/js/chart-test.html b/static/js/chart-test.html new file mode 100644 index 0000000..1229f1a --- /dev/null +++ b/static/js/chart-test.html @@ -0,0 +1,38 @@ + + + + Chart.js Test - Offline + + + +

Chart.js Offline Test

+ + + + + \ No newline at end of file diff --git a/static/js/chart.min.js b/static/js/chart.min.js new file mode 100644 index 0000000..8f69759 --- /dev/null +++ b/static/js/chart.min.js @@ -0,0 +1,13 @@ +/*! + * Chart.js v3.9.1 + * https://www.chartjs.org + * (c) 2022 Chart.js Contributors + * Released under the MIT License + */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).Chart=e()}(this,(function(){"use strict";function t(){}const e=function(){let t=0;return function(){return t++}}();function i(t){return null==t}function s(t){if(Array.isArray&&Array.isArray(t))return!0;const e=Object.prototype.toString.call(t);return"[object"===e.slice(0,7)&&"Array]"===e.slice(-6)}function n(t){return null!==t&&"[object Object]"===Object.prototype.toString.call(t)}const o=t=>("number"==typeof t||t instanceof Number)&&isFinite(+t);function a(t,e){return o(t)?t:e}function r(t,e){return void 0===t?e:t}const l=(t,e)=>"string"==typeof t&&t.endsWith("%")?parseFloat(t)/100:t/e,h=(t,e)=>"string"==typeof t&&t.endsWith("%")?parseFloat(t)/100*e:+t;function c(t,e,i){if(t&&"function"==typeof t.call)return t.apply(i,e)}function d(t,e,i,o){let a,r,l;if(s(t))if(r=t.length,o)for(a=r-1;a>=0;a--)e.call(i,t[a],a);else for(a=0;at,x:t=>t.x,y:t=>t.y};function y(t,e){const i=_[e]||(_[e]=function(t){const e=v(t);return t=>{for(const i of e){if(""===i)break;t=t&&t[i]}return t}}(e));return i(t)}function v(t){const e=t.split("."),i=[];let s="";for(const t of e)s+=t,s.endsWith("\\")?s=s.slice(0,-1)+".":(i.push(s),s="");return i}function w(t){return t.charAt(0).toUpperCase()+t.slice(1)}const M=t=>void 0!==t,k=t=>"function"==typeof t,S=(t,e)=>{if(t.size!==e.size)return!1;for(const i of t)if(!e.has(i))return!1;return!0};function P(t){return"mouseup"===t.type||"click"===t.type||"contextmenu"===t.type}const D=Math.PI,O=2*D,C=O+D,A=Number.POSITIVE_INFINITY,T=D/180,L=D/2,E=D/4,R=2*D/3,I=Math.log10,z=Math.sign;function F(t){const e=Math.round(t);t=N(t,e,t/1e3)?e:t;const i=Math.pow(10,Math.floor(I(t))),s=t/i;return(s<=1?1:s<=2?2:s<=5?5:10)*i}function V(t){const e=[],i=Math.sqrt(t);let s;for(s=1;st-e)).pop(),e}function B(t){return!isNaN(parseFloat(t))&&isFinite(t)}function N(t,e,i){return Math.abs(t-e)=t}function j(t,e,i){let s,n,o;for(s=0,n=t.length;sl&&h=Math.min(e,i)-s&&t<=Math.max(e,i)+s}function tt(t,e,i){i=i||(i=>t[i]1;)s=o+n>>1,i(s)?o=s:n=s;return{lo:o,hi:n}}const et=(t,e,i,s)=>tt(t,i,s?s=>t[s][e]<=i:s=>t[s][e]tt(t,i,(s=>t[s][e]>=i));function st(t,e,i){let s=0,n=t.length;for(;ss&&t[n-1]>i;)n--;return s>0||n{const i="_onData"+w(e),s=t[e];Object.defineProperty(t,e,{configurable:!0,enumerable:!1,value(...e){const n=s.apply(this,e);return t._chartjs.listeners.forEach((t=>{"function"==typeof t[i]&&t[i](...e)})),n}})})))}function at(t,e){const i=t._chartjs;if(!i)return;const s=i.listeners,n=s.indexOf(e);-1!==n&&s.splice(n,1),s.length>0||(nt.forEach((e=>{delete t[e]})),delete t._chartjs)}function rt(t){const e=new Set;let i,s;for(i=0,s=t.length;iArray.prototype.slice.call(t));let n=!1,o=[];return function(...i){o=s(i),n||(n=!0,lt.call(window,(()=>{n=!1,t.apply(e,o)})))}}function ct(t,e){let i;return function(...s){return e?(clearTimeout(i),i=setTimeout(t,e,s)):t.apply(this,s),e}}const dt=t=>"start"===t?"left":"end"===t?"right":"center",ut=(t,e,i)=>"start"===t?e:"end"===t?i:(e+i)/2,ft=(t,e,i,s)=>t===(s?"left":"right")?i:"center"===t?(e+i)/2:e;function gt(t,e,i){const s=e.length;let n=0,o=s;if(t._sorted){const{iScale:a,_parsed:r}=t,l=a.axis,{min:h,max:c,minDefined:d,maxDefined:u}=a.getUserBounds();d&&(n=Z(Math.min(et(r,a.axis,h).lo,i?s:et(e,l,a.getPixelForValue(h)).lo),0,s-1)),o=u?Z(Math.max(et(r,a.axis,c,!0).hi+1,i?0:et(e,l,a.getPixelForValue(c),!0).hi+1),n,s)-n:s-n}return{start:n,count:o}}function pt(t){const{xScale:e,yScale:i,_scaleRanges:s}=t,n={xmin:e.min,xmax:e.max,ymin:i.min,ymax:i.max};if(!s)return t._scaleRanges=n,!0;const o=s.xmin!==e.min||s.xmax!==e.max||s.ymin!==i.min||s.ymax!==i.max;return Object.assign(s,n),o}var mt=new class{constructor(){this._request=null,this._charts=new Map,this._running=!1,this._lastDate=void 0}_notify(t,e,i,s){const n=e.listeners[s],o=e.duration;n.forEach((s=>s({chart:t,initial:e.initial,numSteps:o,currentStep:Math.min(i-e.start,o)})))}_refresh(){this._request||(this._running=!0,this._request=lt.call(window,(()=>{this._update(),this._request=null,this._running&&this._refresh()})))}_update(t=Date.now()){let e=0;this._charts.forEach(((i,s)=>{if(!i.running||!i.items.length)return;const n=i.items;let o,a=n.length-1,r=!1;for(;a>=0;--a)o=n[a],o._active?(o._total>i.duration&&(i.duration=o._total),o.tick(t),r=!0):(n[a]=n[n.length-1],n.pop());r&&(s.draw(),this._notify(s,i,t,"progress")),n.length||(i.running=!1,this._notify(s,i,t,"complete"),i.initial=!1),e+=n.length})),this._lastDate=t,0===e&&(this._running=!1)}_getAnims(t){const e=this._charts;let i=e.get(t);return i||(i={running:!1,initial:!0,items:[],listeners:{complete:[],progress:[]}},e.set(t,i)),i}listen(t,e,i){this._getAnims(t).listeners[e].push(i)}add(t,e){e&&e.length&&this._getAnims(t).items.push(...e)}has(t){return this._getAnims(t).items.length>0}start(t){const e=this._charts.get(t);e&&(e.running=!0,e.start=Date.now(),e.duration=e.items.reduce(((t,e)=>Math.max(t,e._duration)),0),this._refresh())}running(t){if(!this._running)return!1;const e=this._charts.get(t);return!!(e&&e.running&&e.items.length)}stop(t){const e=this._charts.get(t);if(!e||!e.items.length)return;const i=e.items;let s=i.length-1;for(;s>=0;--s)i[s].cancel();e.items=[],this._notify(t,e,Date.now(),"complete")}remove(t){return this._charts.delete(t)}}; +/*! + * @kurkle/color v0.2.1 + * https://github.com/kurkle/color#readme + * (c) 2022 Jukka Kurkela + * Released under the MIT License + */function bt(t){return t+.5|0}const xt=(t,e,i)=>Math.max(Math.min(t,i),e);function _t(t){return xt(bt(2.55*t),0,255)}function yt(t){return xt(bt(255*t),0,255)}function vt(t){return xt(bt(t/2.55)/100,0,1)}function wt(t){return xt(bt(100*t),0,100)}const Mt={0:0,1:1,2:2,3:3,4:4,5:5,6:6,7:7,8:8,9:9,A:10,B:11,C:12,D:13,E:14,F:15,a:10,b:11,c:12,d:13,e:14,f:15},kt=[..."0123456789ABCDEF"],St=t=>kt[15&t],Pt=t=>kt[(240&t)>>4]+kt[15&t],Dt=t=>(240&t)>>4==(15&t);function Ot(t){var e=(t=>Dt(t.r)&&Dt(t.g)&&Dt(t.b)&&Dt(t.a))(t)?St:Pt;return t?"#"+e(t.r)+e(t.g)+e(t.b)+((t,e)=>t<255?e(t):"")(t.a,e):void 0}const Ct=/^(hsla?|hwb|hsv)\(\s*([-+.e\d]+)(?:deg)?[\s,]+([-+.e\d]+)%[\s,]+([-+.e\d]+)%(?:[\s,]+([-+.e\d]+)(%)?)?\s*\)$/;function At(t,e,i){const s=e*Math.min(i,1-i),n=(e,n=(e+t/30)%12)=>i-s*Math.max(Math.min(n-3,9-n,1),-1);return[n(0),n(8),n(4)]}function Tt(t,e,i){const s=(s,n=(s+t/60)%6)=>i-i*e*Math.max(Math.min(n,4-n,1),0);return[s(5),s(3),s(1)]}function Lt(t,e,i){const s=At(t,1,.5);let n;for(e+i>1&&(n=1/(e+i),e*=n,i*=n),n=0;n<3;n++)s[n]*=1-e-i,s[n]+=e;return s}function Et(t){const e=t.r/255,i=t.g/255,s=t.b/255,n=Math.max(e,i,s),o=Math.min(e,i,s),a=(n+o)/2;let r,l,h;return n!==o&&(h=n-o,l=a>.5?h/(2-n-o):h/(n+o),r=function(t,e,i,s,n){return t===n?(e-i)/s+(e>16&255,o>>8&255,255&o]}return t}(),Nt.transparent=[0,0,0,0]);const e=Nt[t.toLowerCase()];return e&&{r:e[0],g:e[1],b:e[2],a:4===e.length?e[3]:255}}const jt=/^rgba?\(\s*([-+.\d]+)(%)?[\s,]+([-+.e\d]+)(%)?[\s,]+([-+.e\d]+)(%)?(?:[\s,/]+([-+.e\d]+)(%)?)?\s*\)$/;const Ht=t=>t<=.0031308?12.92*t:1.055*Math.pow(t,1/2.4)-.055,$t=t=>t<=.04045?t/12.92:Math.pow((t+.055)/1.055,2.4);function Yt(t,e,i){if(t){let s=Et(t);s[e]=Math.max(0,Math.min(s[e]+s[e]*i,0===e?360:1)),s=It(s),t.r=s[0],t.g=s[1],t.b=s[2]}}function Ut(t,e){return t?Object.assign(e||{},t):t}function Xt(t){var e={r:0,g:0,b:0,a:255};return Array.isArray(t)?t.length>=3&&(e={r:t[0],g:t[1],b:t[2],a:255},t.length>3&&(e.a=yt(t[3]))):(e=Ut(t,{r:0,g:0,b:0,a:1})).a=yt(e.a),e}function qt(t){return"r"===t.charAt(0)?function(t){const e=jt.exec(t);let i,s,n,o=255;if(e){if(e[7]!==i){const t=+e[7];o=e[8]?_t(t):xt(255*t,0,255)}return i=+e[1],s=+e[3],n=+e[5],i=255&(e[2]?_t(i):xt(i,0,255)),s=255&(e[4]?_t(s):xt(s,0,255)),n=255&(e[6]?_t(n):xt(n,0,255)),{r:i,g:s,b:n,a:o}}}(t):Ft(t)}class Kt{constructor(t){if(t instanceof Kt)return t;const e=typeof t;let i;var s,n,o;"object"===e?i=Xt(t):"string"===e&&(o=(s=t).length,"#"===s[0]&&(4===o||5===o?n={r:255&17*Mt[s[1]],g:255&17*Mt[s[2]],b:255&17*Mt[s[3]],a:5===o?17*Mt[s[4]]:255}:7!==o&&9!==o||(n={r:Mt[s[1]]<<4|Mt[s[2]],g:Mt[s[3]]<<4|Mt[s[4]],b:Mt[s[5]]<<4|Mt[s[6]],a:9===o?Mt[s[7]]<<4|Mt[s[8]]:255})),i=n||Wt(t)||qt(t)),this._rgb=i,this._valid=!!i}get valid(){return this._valid}get rgb(){var t=Ut(this._rgb);return t&&(t.a=vt(t.a)),t}set rgb(t){this._rgb=Xt(t)}rgbString(){return this._valid?(t=this._rgb)&&(t.a<255?`rgba(${t.r}, ${t.g}, ${t.b}, ${vt(t.a)})`:`rgb(${t.r}, ${t.g}, ${t.b})`):void 0;var t}hexString(){return this._valid?Ot(this._rgb):void 0}hslString(){return this._valid?function(t){if(!t)return;const e=Et(t),i=e[0],s=wt(e[1]),n=wt(e[2]);return t.a<255?`hsla(${i}, ${s}%, ${n}%, ${vt(t.a)})`:`hsl(${i}, ${s}%, ${n}%)`}(this._rgb):void 0}mix(t,e){if(t){const i=this.rgb,s=t.rgb;let n;const o=e===n?.5:e,a=2*o-1,r=i.a-s.a,l=((a*r==-1?a:(a+r)/(1+a*r))+1)/2;n=1-l,i.r=255&l*i.r+n*s.r+.5,i.g=255&l*i.g+n*s.g+.5,i.b=255&l*i.b+n*s.b+.5,i.a=o*i.a+(1-o)*s.a,this.rgb=i}return this}interpolate(t,e){return t&&(this._rgb=function(t,e,i){const s=$t(vt(t.r)),n=$t(vt(t.g)),o=$t(vt(t.b));return{r:yt(Ht(s+i*($t(vt(e.r))-s))),g:yt(Ht(n+i*($t(vt(e.g))-n))),b:yt(Ht(o+i*($t(vt(e.b))-o))),a:t.a+i*(e.a-t.a)}}(this._rgb,t._rgb,e)),this}clone(){return new Kt(this.rgb)}alpha(t){return this._rgb.a=yt(t),this}clearer(t){return this._rgb.a*=1-t,this}greyscale(){const t=this._rgb,e=bt(.3*t.r+.59*t.g+.11*t.b);return t.r=t.g=t.b=e,this}opaquer(t){return this._rgb.a*=1+t,this}negate(){const t=this._rgb;return t.r=255-t.r,t.g=255-t.g,t.b=255-t.b,this}lighten(t){return Yt(this._rgb,2,t),this}darken(t){return Yt(this._rgb,2,-t),this}saturate(t){return Yt(this._rgb,1,t),this}desaturate(t){return Yt(this._rgb,1,-t),this}rotate(t){return function(t,e){var i=Et(t);i[0]=zt(i[0]+e),i=It(i),t.r=i[0],t.g=i[1],t.b=i[2]}(this._rgb,t),this}}function Gt(t){return new Kt(t)}function Zt(t){if(t&&"object"==typeof t){const e=t.toString();return"[object CanvasPattern]"===e||"[object CanvasGradient]"===e}return!1}function Jt(t){return Zt(t)?t:Gt(t)}function Qt(t){return Zt(t)?t:Gt(t).saturate(.5).darken(.1).hexString()}const te=Object.create(null),ee=Object.create(null);function ie(t,e){if(!e)return t;const i=e.split(".");for(let e=0,s=i.length;et.chart.platform.getDevicePixelRatio(),this.elements={},this.events=["mousemove","mouseout","click","touchstart","touchmove"],this.font={family:"'Helvetica Neue', 'Helvetica', 'Arial', sans-serif",size:12,style:"normal",lineHeight:1.2,weight:null},this.hover={},this.hoverBackgroundColor=(t,e)=>Qt(e.backgroundColor),this.hoverBorderColor=(t,e)=>Qt(e.borderColor),this.hoverColor=(t,e)=>Qt(e.color),this.indexAxis="x",this.interaction={mode:"nearest",intersect:!0,includeInvisible:!1},this.maintainAspectRatio=!0,this.onHover=null,this.onClick=null,this.parsing=!0,this.plugins={},this.responsive=!0,this.scale=void 0,this.scales={},this.showLine=!0,this.drawActiveElementsOnTop=!0,this.describe(t)}set(t,e){return se(this,t,e)}get(t){return ie(this,t)}describe(t,e){return se(ee,t,e)}override(t,e){return se(te,t,e)}route(t,e,i,s){const o=ie(this,t),a=ie(this,i),l="_"+e;Object.defineProperties(o,{[l]:{value:o[e],writable:!0},[e]:{enumerable:!0,get(){const t=this[l],e=a[s];return n(t)?Object.assign({},e,t):r(t,e)},set(t){this[l]=t}}})}}({_scriptable:t=>!t.startsWith("on"),_indexable:t=>"events"!==t,hover:{_fallback:"interaction"},interaction:{_scriptable:!1,_indexable:!1}});function oe(){return"undefined"!=typeof window&&"undefined"!=typeof document}function ae(t){let e=t.parentNode;return e&&"[object ShadowRoot]"===e.toString()&&(e=e.host),e}function re(t,e,i){let s;return"string"==typeof t?(s=parseInt(t,10),-1!==t.indexOf("%")&&(s=s/100*e.parentNode[i])):s=t,s}const le=t=>window.getComputedStyle(t,null);function he(t,e){return le(t).getPropertyValue(e)}const ce=["top","right","bottom","left"];function de(t,e,i){const s={};i=i?"-"+i:"";for(let n=0;n<4;n++){const o=ce[n];s[o]=parseFloat(t[e+"-"+o+i])||0}return s.width=s.left+s.right,s.height=s.top+s.bottom,s}function ue(t,e){if("native"in t)return t;const{canvas:i,currentDevicePixelRatio:s}=e,n=le(i),o="border-box"===n.boxSizing,a=de(n,"padding"),r=de(n,"border","width"),{x:l,y:h,box:c}=function(t,e){const i=t.touches,s=i&&i.length?i[0]:t,{offsetX:n,offsetY:o}=s;let a,r,l=!1;if(((t,e,i)=>(t>0||e>0)&&(!i||!i.shadowRoot))(n,o,t.target))a=n,r=o;else{const t=e.getBoundingClientRect();a=s.clientX-t.left,r=s.clientY-t.top,l=!0}return{x:a,y:r,box:l}}(t,i),d=a.left+(c&&r.left),u=a.top+(c&&r.top);let{width:f,height:g}=e;return o&&(f-=a.width+r.width,g-=a.height+r.height),{x:Math.round((l-d)/f*i.width/s),y:Math.round((h-u)/g*i.height/s)}}const fe=t=>Math.round(10*t)/10;function ge(t,e,i,s){const n=le(t),o=de(n,"margin"),a=re(n.maxWidth,t,"clientWidth")||A,r=re(n.maxHeight,t,"clientHeight")||A,l=function(t,e,i){let s,n;if(void 0===e||void 0===i){const o=ae(t);if(o){const t=o.getBoundingClientRect(),a=le(o),r=de(a,"border","width"),l=de(a,"padding");e=t.width-l.width-r.width,i=t.height-l.height-r.height,s=re(a.maxWidth,o,"clientWidth"),n=re(a.maxHeight,o,"clientHeight")}else e=t.clientWidth,i=t.clientHeight}return{width:e,height:i,maxWidth:s||A,maxHeight:n||A}}(t,e,i);let{width:h,height:c}=l;if("content-box"===n.boxSizing){const t=de(n,"border","width"),e=de(n,"padding");h-=e.width+t.width,c-=e.height+t.height}return h=Math.max(0,h-o.width),c=Math.max(0,s?Math.floor(h/s):c-o.height),h=fe(Math.min(h,a,l.maxWidth)),c=fe(Math.min(c,r,l.maxHeight)),h&&!c&&(c=fe(h/2)),{width:h,height:c}}function pe(t,e,i){const s=e||1,n=Math.floor(t.height*s),o=Math.floor(t.width*s);t.height=n/s,t.width=o/s;const a=t.canvas;return a.style&&(i||!a.style.height&&!a.style.width)&&(a.style.height=`${t.height}px`,a.style.width=`${t.width}px`),(t.currentDevicePixelRatio!==s||a.height!==n||a.width!==o)&&(t.currentDevicePixelRatio=s,a.height=n,a.width=o,t.ctx.setTransform(s,0,0,s,0,0),!0)}const me=function(){let t=!1;try{const e={get passive(){return t=!0,!1}};window.addEventListener("test",null,e),window.removeEventListener("test",null,e)}catch(t){}return t}();function be(t,e){const i=he(t,e),s=i&&i.match(/^(\d+)(\.\d+)?px$/);return s?+s[1]:void 0}function xe(t){return!t||i(t.size)||i(t.family)?null:(t.style?t.style+" ":"")+(t.weight?t.weight+" ":"")+t.size+"px "+t.family}function _e(t,e,i,s,n){let o=e[n];return o||(o=e[n]=t.measureText(n).width,i.push(n)),o>s&&(s=o),s}function ye(t,e,i,n){let o=(n=n||{}).data=n.data||{},a=n.garbageCollect=n.garbageCollect||[];n.font!==e&&(o=n.data={},a=n.garbageCollect=[],n.font=e),t.save(),t.font=e;let r=0;const l=i.length;let h,c,d,u,f;for(h=0;hi.length){for(h=0;h0&&t.stroke()}}function Se(t,e,i){return i=i||.5,!e||t&&t.x>e.left-i&&t.xe.top-i&&t.y0&&""!==r.strokeColor;let c,d;for(t.save(),t.font=a.string,function(t,e){e.translation&&t.translate(e.translation[0],e.translation[1]);i(e.rotation)||t.rotate(e.rotation);e.color&&(t.fillStyle=e.color);e.textAlign&&(t.textAlign=e.textAlign);e.textBaseline&&(t.textBaseline=e.textBaseline)}(t,r),c=0;ct[0])){M(s)||(s=$e("_fallback",t));const o={[Symbol.toStringTag]:"Object",_cacheable:!0,_scopes:t,_rootScopes:i,_fallback:s,_getTarget:n,override:n=>Ee([n,...t],e,i,s)};return new Proxy(o,{deleteProperty:(e,i)=>(delete e[i],delete e._keys,delete t[0][i],!0),get:(i,s)=>Ve(i,s,(()=>function(t,e,i,s){let n;for(const o of e)if(n=$e(ze(o,t),i),M(n))return Fe(t,n)?je(i,s,t,n):n}(s,e,t,i))),getOwnPropertyDescriptor:(t,e)=>Reflect.getOwnPropertyDescriptor(t._scopes[0],e),getPrototypeOf:()=>Reflect.getPrototypeOf(t[0]),has:(t,e)=>Ye(t).includes(e),ownKeys:t=>Ye(t),set(t,e,i){const s=t._storage||(t._storage=n());return t[e]=s[e]=i,delete t._keys,!0}})}function Re(t,e,i,o){const a={_cacheable:!1,_proxy:t,_context:e,_subProxy:i,_stack:new Set,_descriptors:Ie(t,o),setContext:e=>Re(t,e,i,o),override:s=>Re(t.override(s),e,i,o)};return new Proxy(a,{deleteProperty:(e,i)=>(delete e[i],delete t[i],!0),get:(t,e,i)=>Ve(t,e,(()=>function(t,e,i){const{_proxy:o,_context:a,_subProxy:r,_descriptors:l}=t;let h=o[e];k(h)&&l.isScriptable(e)&&(h=function(t,e,i,s){const{_proxy:n,_context:o,_subProxy:a,_stack:r}=i;if(r.has(t))throw new Error("Recursion detected: "+Array.from(r).join("->")+"->"+t);r.add(t),e=e(o,a||s),r.delete(t),Fe(t,e)&&(e=je(n._scopes,n,t,e));return e}(e,h,t,i));s(h)&&h.length&&(h=function(t,e,i,s){const{_proxy:o,_context:a,_subProxy:r,_descriptors:l}=i;if(M(a.index)&&s(t))e=e[a.index%e.length];else if(n(e[0])){const i=e,s=o._scopes.filter((t=>t!==i));e=[];for(const n of i){const i=je(s,o,t,n);e.push(Re(i,a,r&&r[t],l))}}return e}(e,h,t,l.isIndexable));Fe(e,h)&&(h=Re(h,a,r&&r[e],l));return h}(t,e,i))),getOwnPropertyDescriptor:(e,i)=>e._descriptors.allKeys?Reflect.has(t,i)?{enumerable:!0,configurable:!0}:void 0:Reflect.getOwnPropertyDescriptor(t,i),getPrototypeOf:()=>Reflect.getPrototypeOf(t),has:(e,i)=>Reflect.has(t,i),ownKeys:()=>Reflect.ownKeys(t),set:(e,i,s)=>(t[i]=s,delete e[i],!0)})}function Ie(t,e={scriptable:!0,indexable:!0}){const{_scriptable:i=e.scriptable,_indexable:s=e.indexable,_allKeys:n=e.allKeys}=t;return{allKeys:n,scriptable:i,indexable:s,isScriptable:k(i)?i:()=>i,isIndexable:k(s)?s:()=>s}}const ze=(t,e)=>t?t+w(e):e,Fe=(t,e)=>n(e)&&"adapters"!==t&&(null===Object.getPrototypeOf(e)||e.constructor===Object);function Ve(t,e,i){if(Object.prototype.hasOwnProperty.call(t,e))return t[e];const s=i();return t[e]=s,s}function Be(t,e,i){return k(t)?t(e,i):t}const Ne=(t,e)=>!0===t?e:"string"==typeof t?y(e,t):void 0;function We(t,e,i,s,n){for(const o of e){const e=Ne(i,o);if(e){t.add(e);const o=Be(e._fallback,i,n);if(M(o)&&o!==i&&o!==s)return o}else if(!1===e&&M(s)&&i!==s)return null}return!1}function je(t,e,i,o){const a=e._rootScopes,r=Be(e._fallback,i,o),l=[...t,...a],h=new Set;h.add(o);let c=He(h,l,i,r||i,o);return null!==c&&((!M(r)||r===i||(c=He(h,l,r,c,o),null!==c))&&Ee(Array.from(h),[""],a,r,(()=>function(t,e,i){const o=t._getTarget();e in o||(o[e]={});const a=o[e];if(s(a)&&n(i))return i;return a}(e,i,o))))}function He(t,e,i,s,n){for(;i;)i=We(t,e,i,s,n);return i}function $e(t,e){for(const i of e){if(!i)continue;const e=i[t];if(M(e))return e}}function Ye(t){let e=t._keys;return e||(e=t._keys=function(t){const e=new Set;for(const i of t)for(const t of Object.keys(i).filter((t=>!t.startsWith("_"))))e.add(t);return Array.from(e)}(t._scopes)),e}function Ue(t,e,i,s){const{iScale:n}=t,{key:o="r"}=this._parsing,a=new Array(s);let r,l,h,c;for(r=0,l=s;re"x"===t?"y":"x";function Ge(t,e,i,s){const n=t.skip?e:t,o=e,a=i.skip?e:i,r=X(o,n),l=X(a,o);let h=r/(r+l),c=l/(r+l);h=isNaN(h)?0:h,c=isNaN(c)?0:c;const d=s*h,u=s*c;return{previous:{x:o.x-d*(a.x-n.x),y:o.y-d*(a.y-n.y)},next:{x:o.x+u*(a.x-n.x),y:o.y+u*(a.y-n.y)}}}function Ze(t,e="x"){const i=Ke(e),s=t.length,n=Array(s).fill(0),o=Array(s);let a,r,l,h=qe(t,0);for(a=0;a!t.skip))),"monotone"===e.cubicInterpolationMode)Ze(t,n);else{let i=s?t[t.length-1]:t[0];for(o=0,a=t.length;o0===t||1===t,ei=(t,e,i)=>-Math.pow(2,10*(t-=1))*Math.sin((t-e)*O/i),ii=(t,e,i)=>Math.pow(2,-10*t)*Math.sin((t-e)*O/i)+1,si={linear:t=>t,easeInQuad:t=>t*t,easeOutQuad:t=>-t*(t-2),easeInOutQuad:t=>(t/=.5)<1?.5*t*t:-.5*(--t*(t-2)-1),easeInCubic:t=>t*t*t,easeOutCubic:t=>(t-=1)*t*t+1,easeInOutCubic:t=>(t/=.5)<1?.5*t*t*t:.5*((t-=2)*t*t+2),easeInQuart:t=>t*t*t*t,easeOutQuart:t=>-((t-=1)*t*t*t-1),easeInOutQuart:t=>(t/=.5)<1?.5*t*t*t*t:-.5*((t-=2)*t*t*t-2),easeInQuint:t=>t*t*t*t*t,easeOutQuint:t=>(t-=1)*t*t*t*t+1,easeInOutQuint:t=>(t/=.5)<1?.5*t*t*t*t*t:.5*((t-=2)*t*t*t*t+2),easeInSine:t=>1-Math.cos(t*L),easeOutSine:t=>Math.sin(t*L),easeInOutSine:t=>-.5*(Math.cos(D*t)-1),easeInExpo:t=>0===t?0:Math.pow(2,10*(t-1)),easeOutExpo:t=>1===t?1:1-Math.pow(2,-10*t),easeInOutExpo:t=>ti(t)?t:t<.5?.5*Math.pow(2,10*(2*t-1)):.5*(2-Math.pow(2,-10*(2*t-1))),easeInCirc:t=>t>=1?t:-(Math.sqrt(1-t*t)-1),easeOutCirc:t=>Math.sqrt(1-(t-=1)*t),easeInOutCirc:t=>(t/=.5)<1?-.5*(Math.sqrt(1-t*t)-1):.5*(Math.sqrt(1-(t-=2)*t)+1),easeInElastic:t=>ti(t)?t:ei(t,.075,.3),easeOutElastic:t=>ti(t)?t:ii(t,.075,.3),easeInOutElastic(t){const e=.1125;return ti(t)?t:t<.5?.5*ei(2*t,e,.45):.5+.5*ii(2*t-1,e,.45)},easeInBack(t){const e=1.70158;return t*t*((e+1)*t-e)},easeOutBack(t){const e=1.70158;return(t-=1)*t*((e+1)*t+e)+1},easeInOutBack(t){let e=1.70158;return(t/=.5)<1?t*t*((1+(e*=1.525))*t-e)*.5:.5*((t-=2)*t*((1+(e*=1.525))*t+e)+2)},easeInBounce:t=>1-si.easeOutBounce(1-t),easeOutBounce(t){const e=7.5625,i=2.75;return t<1/i?e*t*t:t<2/i?e*(t-=1.5/i)*t+.75:t<2.5/i?e*(t-=2.25/i)*t+.9375:e*(t-=2.625/i)*t+.984375},easeInOutBounce:t=>t<.5?.5*si.easeInBounce(2*t):.5*si.easeOutBounce(2*t-1)+.5};function ni(t,e,i,s){return{x:t.x+i*(e.x-t.x),y:t.y+i*(e.y-t.y)}}function oi(t,e,i,s){return{x:t.x+i*(e.x-t.x),y:"middle"===s?i<.5?t.y:e.y:"after"===s?i<1?t.y:e.y:i>0?e.y:t.y}}function ai(t,e,i,s){const n={x:t.cp2x,y:t.cp2y},o={x:e.cp1x,y:e.cp1y},a=ni(t,n,i),r=ni(n,o,i),l=ni(o,e,i),h=ni(a,r,i),c=ni(r,l,i);return ni(h,c,i)}const ri=new Map;function li(t,e,i){return function(t,e){e=e||{};const i=t+JSON.stringify(e);let s=ri.get(i);return s||(s=new Intl.NumberFormat(t,e),ri.set(i,s)),s}(e,i).format(t)}const hi=new RegExp(/^(normal|(\d+(?:\.\d+)?)(px|em|%)?)$/),ci=new RegExp(/^(normal|italic|initial|inherit|unset|(oblique( -?[0-9]?[0-9]deg)?))$/);function di(t,e){const i=(""+t).match(hi);if(!i||"normal"===i[1])return 1.2*e;switch(t=+i[2],i[3]){case"px":return t;case"%":t/=100}return e*t}function ui(t,e){const i={},s=n(e),o=s?Object.keys(e):e,a=n(t)?s?i=>r(t[i],t[e[i]]):e=>t[e]:()=>t;for(const t of o)i[t]=+a(t)||0;return i}function fi(t){return ui(t,{top:"y",right:"x",bottom:"y",left:"x"})}function gi(t){return ui(t,["topLeft","topRight","bottomLeft","bottomRight"])}function pi(t){const e=fi(t);return e.width=e.left+e.right,e.height=e.top+e.bottom,e}function mi(t,e){t=t||{},e=e||ne.font;let i=r(t.size,e.size);"string"==typeof i&&(i=parseInt(i,10));let s=r(t.style,e.style);s&&!(""+s).match(ci)&&(console.warn('Invalid font style specified: "'+s+'"'),s="");const n={family:r(t.family,e.family),lineHeight:di(r(t.lineHeight,e.lineHeight),i),size:i,style:s,weight:r(t.weight,e.weight),string:""};return n.string=xe(n),n}function bi(t,e,i,n){let o,a,r,l=!0;for(o=0,a=t.length;oi&&0===t?0:t+e;return{min:a(s,-Math.abs(o)),max:a(n,o)}}function _i(t,e){return Object.assign(Object.create(t),e)}function yi(t,e,i){return t?function(t,e){return{x:i=>t+t+e-i,setWidth(t){e=t},textAlign:t=>"center"===t?t:"right"===t?"left":"right",xPlus:(t,e)=>t-e,leftForLtr:(t,e)=>t-e}}(e,i):{x:t=>t,setWidth(t){},textAlign:t=>t,xPlus:(t,e)=>t+e,leftForLtr:(t,e)=>t}}function vi(t,e){let i,s;"ltr"!==e&&"rtl"!==e||(i=t.canvas.style,s=[i.getPropertyValue("direction"),i.getPropertyPriority("direction")],i.setProperty("direction",e,"important"),t.prevTextDirection=s)}function wi(t,e){void 0!==e&&(delete t.prevTextDirection,t.canvas.style.setProperty("direction",e[0],e[1]))}function Mi(t){return"angle"===t?{between:G,compare:q,normalize:K}:{between:Q,compare:(t,e)=>t-e,normalize:t=>t}}function ki({start:t,end:e,count:i,loop:s,style:n}){return{start:t%i,end:e%i,loop:s&&(e-t+1)%i==0,style:n}}function Si(t,e,i){if(!i)return[t];const{property:s,start:n,end:o}=i,a=e.length,{compare:r,between:l,normalize:h}=Mi(s),{start:c,end:d,loop:u,style:f}=function(t,e,i){const{property:s,start:n,end:o}=i,{between:a,normalize:r}=Mi(s),l=e.length;let h,c,{start:d,end:u,loop:f}=t;if(f){for(d+=l,u+=l,h=0,c=l;hx||l(n,b,p)&&0!==r(n,b),v=()=>!x||0===r(o,p)||l(o,b,p);for(let t=c,i=c;t<=d;++t)m=e[t%a],m.skip||(p=h(m[s]),p!==b&&(x=l(p,n,o),null===_&&y()&&(_=0===r(p,n)?t:i),null!==_&&v()&&(g.push(ki({start:_,end:t,loop:u,count:a,style:f})),_=null),i=t,b=p));return null!==_&&g.push(ki({start:_,end:d,loop:u,count:a,style:f})),g}function Pi(t,e){const i=[],s=t.segments;for(let n=0;nn&&t[o%e].skip;)o--;return o%=e,{start:n,end:o}}(i,n,o,s);if(!0===s)return Oi(t,[{start:a,end:r,loop:o}],i,e);return Oi(t,function(t,e,i,s){const n=t.length,o=[];let a,r=e,l=t[e];for(a=e+1;a<=i;++a){const i=t[a%n];i.skip||i.stop?l.skip||(s=!1,o.push({start:e%n,end:(a-1)%n,loop:s}),e=r=i.stop?a:null):(r=a,l.skip&&(e=a)),l=i}return null!==r&&o.push({start:e%n,end:r%n,loop:s}),o}(i,a,r{t[a](e[i],n)&&(o.push({element:t,datasetIndex:s,index:l}),r=r||t.inRange(e.x,e.y,n))})),s&&!r?[]:o}var Vi={evaluateInteractionItems:Ei,modes:{index(t,e,i,s){const n=ue(e,t),o=i.axis||"x",a=i.includeInvisible||!1,r=i.intersect?Ri(t,n,o,s,a):zi(t,n,o,!1,s,a),l=[];return r.length?(t.getSortedVisibleDatasetMetas().forEach((t=>{const e=r[0].index,i=t.data[e];i&&!i.skip&&l.push({element:i,datasetIndex:t.index,index:e})})),l):[]},dataset(t,e,i,s){const n=ue(e,t),o=i.axis||"xy",a=i.includeInvisible||!1;let r=i.intersect?Ri(t,n,o,s,a):zi(t,n,o,!1,s,a);if(r.length>0){const e=r[0].datasetIndex,i=t.getDatasetMeta(e).data;r=[];for(let t=0;tRi(t,ue(e,t),i.axis||"xy",s,i.includeInvisible||!1),nearest(t,e,i,s){const n=ue(e,t),o=i.axis||"xy",a=i.includeInvisible||!1;return zi(t,n,o,i.intersect,s,a)},x:(t,e,i,s)=>Fi(t,ue(e,t),"x",i.intersect,s),y:(t,e,i,s)=>Fi(t,ue(e,t),"y",i.intersect,s)}};const Bi=["left","top","right","bottom"];function Ni(t,e){return t.filter((t=>t.pos===e))}function Wi(t,e){return t.filter((t=>-1===Bi.indexOf(t.pos)&&t.box.axis===e))}function ji(t,e){return t.sort(((t,i)=>{const s=e?i:t,n=e?t:i;return s.weight===n.weight?s.index-n.index:s.weight-n.weight}))}function Hi(t,e){const i=function(t){const e={};for(const i of t){const{stack:t,pos:s,stackWeight:n}=i;if(!t||!Bi.includes(s))continue;const o=e[t]||(e[t]={count:0,placed:0,weight:0,size:0});o.count++,o.weight+=n}return e}(t),{vBoxMaxWidth:s,hBoxMaxHeight:n}=e;let o,a,r;for(o=0,a=t.length;o{s[t]=Math.max(e[t],i[t])})),s}return s(t?["left","right"]:["top","bottom"])}function qi(t,e,i,s){const n=[];let o,a,r,l,h,c;for(o=0,a=t.length,h=0;ot.box.fullSize)),!0),s=ji(Ni(e,"left"),!0),n=ji(Ni(e,"right")),o=ji(Ni(e,"top"),!0),a=ji(Ni(e,"bottom")),r=Wi(e,"x"),l=Wi(e,"y");return{fullSize:i,leftAndTop:s.concat(o),rightAndBottom:n.concat(l).concat(a).concat(r),chartArea:Ni(e,"chartArea"),vertical:s.concat(n).concat(l),horizontal:o.concat(a).concat(r)}}(t.boxes),l=r.vertical,h=r.horizontal;d(t.boxes,(t=>{"function"==typeof t.beforeLayout&&t.beforeLayout()}));const c=l.reduce(((t,e)=>e.box.options&&!1===e.box.options.display?t:t+1),0)||1,u=Object.freeze({outerWidth:e,outerHeight:i,padding:n,availableWidth:o,availableHeight:a,vBoxMaxWidth:o/2/c,hBoxMaxHeight:a/2}),f=Object.assign({},n);Yi(f,pi(s));const g=Object.assign({maxPadding:f,w:o,h:a,x:n.left,y:n.top},n),p=Hi(l.concat(h),u);qi(r.fullSize,g,u,p),qi(l,g,u,p),qi(h,g,u,p)&&qi(l,g,u,p),function(t){const e=t.maxPadding;function i(i){const s=Math.max(e[i]-t[i],0);return t[i]+=s,s}t.y+=i("top"),t.x+=i("left"),i("right"),i("bottom")}(g),Gi(r.leftAndTop,g,u,p),g.x+=g.w,g.y+=g.h,Gi(r.rightAndBottom,g,u,p),t.chartArea={left:g.left,top:g.top,right:g.left+g.w,bottom:g.top+g.h,height:g.h,width:g.w},d(r.chartArea,(e=>{const i=e.box;Object.assign(i,t.chartArea),i.update(g.w,g.h,{left:0,top:0,right:0,bottom:0})}))}};class Ji{acquireContext(t,e){}releaseContext(t){return!1}addEventListener(t,e,i){}removeEventListener(t,e,i){}getDevicePixelRatio(){return 1}getMaximumSize(t,e,i,s){return e=Math.max(0,e||t.width),i=i||t.height,{width:e,height:Math.max(0,s?Math.floor(e/s):i)}}isAttached(t){return!0}updateConfig(t){}}class Qi extends Ji{acquireContext(t){return t&&t.getContext&&t.getContext("2d")||null}updateConfig(t){t.options.animation=!1}}const ts={touchstart:"mousedown",touchmove:"mousemove",touchend:"mouseup",pointerenter:"mouseenter",pointerdown:"mousedown",pointermove:"mousemove",pointerup:"mouseup",pointerleave:"mouseout",pointerout:"mouseout"},es=t=>null===t||""===t;const is=!!me&&{passive:!0};function ss(t,e,i){t.canvas.removeEventListener(e,i,is)}function ns(t,e){for(const i of t)if(i===e||i.contains(e))return!0}function os(t,e,i){const s=t.canvas,n=new MutationObserver((t=>{let e=!1;for(const i of t)e=e||ns(i.addedNodes,s),e=e&&!ns(i.removedNodes,s);e&&i()}));return n.observe(document,{childList:!0,subtree:!0}),n}function as(t,e,i){const s=t.canvas,n=new MutationObserver((t=>{let e=!1;for(const i of t)e=e||ns(i.removedNodes,s),e=e&&!ns(i.addedNodes,s);e&&i()}));return n.observe(document,{childList:!0,subtree:!0}),n}const rs=new Map;let ls=0;function hs(){const t=window.devicePixelRatio;t!==ls&&(ls=t,rs.forEach(((e,i)=>{i.currentDevicePixelRatio!==t&&e()})))}function cs(t,e,i){const s=t.canvas,n=s&&ae(s);if(!n)return;const o=ht(((t,e)=>{const s=n.clientWidth;i(t,e),s{const e=t[0],i=e.contentRect.width,s=e.contentRect.height;0===i&&0===s||o(i,s)}));return a.observe(n),function(t,e){rs.size||window.addEventListener("resize",hs),rs.set(t,e)}(t,o),a}function ds(t,e,i){i&&i.disconnect(),"resize"===e&&function(t){rs.delete(t),rs.size||window.removeEventListener("resize",hs)}(t)}function us(t,e,i){const s=t.canvas,n=ht((e=>{null!==t.ctx&&i(function(t,e){const i=ts[t.type]||t.type,{x:s,y:n}=ue(t,e);return{type:i,chart:e,native:t,x:void 0!==s?s:null,y:void 0!==n?n:null}}(e,t))}),t,(t=>{const e=t[0];return[e,e.offsetX,e.offsetY]}));return function(t,e,i){t.addEventListener(e,i,is)}(s,e,n),n}class fs extends Ji{acquireContext(t,e){const i=t&&t.getContext&&t.getContext("2d");return i&&i.canvas===t?(function(t,e){const i=t.style,s=t.getAttribute("height"),n=t.getAttribute("width");if(t.$chartjs={initial:{height:s,width:n,style:{display:i.display,height:i.height,width:i.width}}},i.display=i.display||"block",i.boxSizing=i.boxSizing||"border-box",es(n)){const e=be(t,"width");void 0!==e&&(t.width=e)}if(es(s))if(""===t.style.height)t.height=t.width/(e||2);else{const e=be(t,"height");void 0!==e&&(t.height=e)}}(t,e),i):null}releaseContext(t){const e=t.canvas;if(!e.$chartjs)return!1;const s=e.$chartjs.initial;["height","width"].forEach((t=>{const n=s[t];i(n)?e.removeAttribute(t):e.setAttribute(t,n)}));const n=s.style||{};return Object.keys(n).forEach((t=>{e.style[t]=n[t]})),e.width=e.width,delete e.$chartjs,!0}addEventListener(t,e,i){this.removeEventListener(t,e);const s=t.$proxies||(t.$proxies={}),n={attach:os,detach:as,resize:cs}[e]||us;s[e]=n(t,e,i)}removeEventListener(t,e){const i=t.$proxies||(t.$proxies={}),s=i[e];if(!s)return;({attach:ds,detach:ds,resize:ds}[e]||ss)(t,e,s),i[e]=void 0}getDevicePixelRatio(){return window.devicePixelRatio}getMaximumSize(t,e,i,s){return ge(t,e,i,s)}isAttached(t){const e=ae(t);return!(!e||!e.isConnected)}}function gs(t){return!oe()||"undefined"!=typeof OffscreenCanvas&&t instanceof OffscreenCanvas?Qi:fs}var ps=Object.freeze({__proto__:null,_detectPlatform:gs,BasePlatform:Ji,BasicPlatform:Qi,DomPlatform:fs});const ms="transparent",bs={boolean:(t,e,i)=>i>.5?e:t,color(t,e,i){const s=Jt(t||ms),n=s.valid&&Jt(e||ms);return n&&n.valid?n.mix(s,i).hexString():e},number:(t,e,i)=>t+(e-t)*i};class xs{constructor(t,e,i,s){const n=e[i];s=bi([t.to,s,n,t.from]);const o=bi([t.from,n,s]);this._active=!0,this._fn=t.fn||bs[t.type||typeof o],this._easing=si[t.easing]||si.linear,this._start=Math.floor(Date.now()+(t.delay||0)),this._duration=this._total=Math.floor(t.duration),this._loop=!!t.loop,this._target=e,this._prop=i,this._from=o,this._to=s,this._promises=void 0}active(){return this._active}update(t,e,i){if(this._active){this._notify(!1);const s=this._target[this._prop],n=i-this._start,o=this._duration-n;this._start=i,this._duration=Math.floor(Math.max(o,t.duration)),this._total+=n,this._loop=!!t.loop,this._to=bi([t.to,e,s,t.from]),this._from=bi([t.from,s,e])}}cancel(){this._active&&(this.tick(Date.now()),this._active=!1,this._notify(!1))}tick(t){const e=t-this._start,i=this._duration,s=this._prop,n=this._from,o=this._loop,a=this._to;let r;if(this._active=n!==a&&(o||e1?2-r:r,r=this._easing(Math.min(1,Math.max(0,r))),this._target[s]=this._fn(n,a,r))}wait(){const t=this._promises||(this._promises=[]);return new Promise(((e,i)=>{t.push({res:e,rej:i})}))}_notify(t){const e=t?"res":"rej",i=this._promises||[];for(let t=0;t"onProgress"!==t&&"onComplete"!==t&&"fn"!==t}),ne.set("animations",{colors:{type:"color",properties:["color","borderColor","backgroundColor"]},numbers:{type:"number",properties:["x","y","borderWidth","radius","tension"]}}),ne.describe("animations",{_fallback:"animation"}),ne.set("transitions",{active:{animation:{duration:400}},resize:{animation:{duration:0}},show:{animations:{colors:{from:"transparent"},visible:{type:"boolean",duration:0}}},hide:{animations:{colors:{to:"transparent"},visible:{type:"boolean",easing:"linear",fn:t=>0|t}}}});class ys{constructor(t,e){this._chart=t,this._properties=new Map,this.configure(e)}configure(t){if(!n(t))return;const e=this._properties;Object.getOwnPropertyNames(t).forEach((i=>{const o=t[i];if(!n(o))return;const a={};for(const t of _s)a[t]=o[t];(s(o.properties)&&o.properties||[i]).forEach((t=>{t!==i&&e.has(t)||e.set(t,a)}))}))}_animateOptions(t,e){const i=e.options,s=function(t,e){if(!e)return;let i=t.options;if(!i)return void(t.options=e);i.$shared&&(t.options=i=Object.assign({},i,{$shared:!1,$animations:{}}));return i}(t,i);if(!s)return[];const n=this._createAnimations(s,i);return i.$shared&&function(t,e){const i=[],s=Object.keys(e);for(let e=0;e{t.options=i}),(()=>{})),n}_createAnimations(t,e){const i=this._properties,s=[],n=t.$animations||(t.$animations={}),o=Object.keys(e),a=Date.now();let r;for(r=o.length-1;r>=0;--r){const l=o[r];if("$"===l.charAt(0))continue;if("options"===l){s.push(...this._animateOptions(t,e));continue}const h=e[l];let c=n[l];const d=i.get(l);if(c){if(d&&c.active()){c.update(d,h,a);continue}c.cancel()}d&&d.duration?(n[l]=c=new xs(d,t,l,h),s.push(c)):t[l]=h}return s}update(t,e){if(0===this._properties.size)return void Object.assign(t,e);const i=this._createAnimations(t,e);return i.length?(mt.add(this._chart,i),!0):void 0}}function vs(t,e){const i=t&&t.options||{},s=i.reverse,n=void 0===i.min?e:0,o=void 0===i.max?e:0;return{start:s?o:n,end:s?n:o}}function ws(t,e){const i=[],s=t._getSortedDatasetMetas(e);let n,o;for(n=0,o=s.length;n0||!i&&e<0)return n.index}return null}function Ds(t,e){const{chart:i,_cachedMeta:s}=t,n=i._stacks||(i._stacks={}),{iScale:o,vScale:a,index:r}=s,l=o.axis,h=a.axis,c=function(t,e,i){return`${t.id}.${e.id}.${i.stack||i.type}`}(o,a,s),d=e.length;let u;for(let t=0;ti[t].axis===e)).shift()}function Cs(t,e){const i=t.controller.index,s=t.vScale&&t.vScale.axis;if(s){e=e||t._parsed;for(const t of e){const e=t._stacks;if(!e||void 0===e[s]||void 0===e[s][i])return;delete e[s][i]}}}const As=t=>"reset"===t||"none"===t,Ts=(t,e)=>e?t:Object.assign({},t);class Ls{constructor(t,e){this.chart=t,this._ctx=t.ctx,this.index=e,this._cachedDataOpts={},this._cachedMeta=this.getMeta(),this._type=this._cachedMeta.type,this.options=void 0,this._parsing=!1,this._data=void 0,this._objectData=void 0,this._sharedOptions=void 0,this._drawStart=void 0,this._drawCount=void 0,this.enableOptionSharing=!1,this.supportsDecimation=!1,this.$context=void 0,this._syncList=[],this.initialize()}initialize(){const t=this._cachedMeta;this.configure(),this.linkScales(),t._stacked=ks(t.vScale,t),this.addElements()}updateIndex(t){this.index!==t&&Cs(this._cachedMeta),this.index=t}linkScales(){const t=this.chart,e=this._cachedMeta,i=this.getDataset(),s=(t,e,i,s)=>"x"===t?e:"r"===t?s:i,n=e.xAxisID=r(i.xAxisID,Os(t,"x")),o=e.yAxisID=r(i.yAxisID,Os(t,"y")),a=e.rAxisID=r(i.rAxisID,Os(t,"r")),l=e.indexAxis,h=e.iAxisID=s(l,n,o,a),c=e.vAxisID=s(l,o,n,a);e.xScale=this.getScaleForId(n),e.yScale=this.getScaleForId(o),e.rScale=this.getScaleForId(a),e.iScale=this.getScaleForId(h),e.vScale=this.getScaleForId(c)}getDataset(){return this.chart.data.datasets[this.index]}getMeta(){return this.chart.getDatasetMeta(this.index)}getScaleForId(t){return this.chart.scales[t]}_getOtherScale(t){const e=this._cachedMeta;return t===e.iScale?e.vScale:e.iScale}reset(){this._update("reset")}_destroy(){const t=this._cachedMeta;this._data&&at(this._data,this),t._stacked&&Cs(t)}_dataCheck(){const t=this.getDataset(),e=t.data||(t.data=[]),i=this._data;if(n(e))this._data=function(t){const e=Object.keys(t),i=new Array(e.length);let s,n,o;for(s=0,n=e.length;s0&&i._parsed[t-1];if(!1===this._parsing)i._parsed=o,i._sorted=!0,d=o;else{d=s(o[t])?this.parseArrayData(i,o,t,e):n(o[t])?this.parseObjectData(i,o,t,e):this.parsePrimitiveData(i,o,t,e);const a=()=>null===c[l]||f&&c[l]t&&!e.hidden&&e._stacked&&{keys:ws(i,!0),values:null})(e,i,this.chart),h={min:Number.POSITIVE_INFINITY,max:Number.NEGATIVE_INFINITY},{min:c,max:d}=function(t){const{min:e,max:i,minDefined:s,maxDefined:n}=t.getUserBounds();return{min:s?e:Number.NEGATIVE_INFINITY,max:n?i:Number.POSITIVE_INFINITY}}(r);let u,f;function g(){f=s[u];const e=f[r.axis];return!o(f[t.axis])||c>e||d=0;--u)if(!g()){this.updateRangeFromParsed(h,t,f,l);break}return h}getAllParsedValues(t){const e=this._cachedMeta._parsed,i=[];let s,n,a;for(s=0,n=e.length;s=0&&tthis.getContext(i,s)),c);return f.$shared&&(f.$shared=r,n[o]=Object.freeze(Ts(f,r))),f}_resolveAnimations(t,e,i){const s=this.chart,n=this._cachedDataOpts,o=`animation-${e}`,a=n[o];if(a)return a;let r;if(!1!==s.options.animation){const s=this.chart.config,n=s.datasetAnimationScopeKeys(this._type,e),o=s.getOptionScopes(this.getDataset(),n);r=s.createResolver(o,this.getContext(t,i,e))}const l=new ys(s,r&&r.animations);return r&&r._cacheable&&(n[o]=Object.freeze(l)),l}getSharedOptions(t){if(t.$shared)return this._sharedOptions||(this._sharedOptions=Object.assign({},t))}includeOptions(t,e){return!e||As(t)||this.chart._animationsDisabled}_getSharedOptions(t,e){const i=this.resolveDataElementOptions(t,e),s=this._sharedOptions,n=this.getSharedOptions(i),o=this.includeOptions(e,n)||n!==s;return this.updateSharedOptions(n,e,i),{sharedOptions:n,includeOptions:o}}updateElement(t,e,i,s){As(s)?Object.assign(t,i):this._resolveAnimations(e,s).update(t,i)}updateSharedOptions(t,e,i){t&&!As(e)&&this._resolveAnimations(void 0,e).update(t,i)}_setStyle(t,e,i,s){t.active=s;const n=this.getStyle(e,s);this._resolveAnimations(e,i,s).update(t,{options:!s&&this.getSharedOptions(n)||n})}removeHoverStyle(t,e,i){this._setStyle(t,i,"active",!1)}setHoverStyle(t,e,i){this._setStyle(t,i,"active",!0)}_removeDatasetHoverStyle(){const t=this._cachedMeta.dataset;t&&this._setStyle(t,void 0,"active",!1)}_setDatasetHoverStyle(){const t=this._cachedMeta.dataset;t&&this._setStyle(t,void 0,"active",!0)}_resyncElements(t){const e=this._data,i=this._cachedMeta.data;for(const[t,e,i]of this._syncList)this[t](e,i);this._syncList=[];const s=i.length,n=e.length,o=Math.min(n,s);o&&this.parse(0,o),n>s?this._insertElements(s,n-s,t):n{for(t.length+=e,a=t.length-1;a>=o;a--)t[a]=t[a-e]};for(r(n),a=t;a{s[t]=i[t]&&i[t].active()?i[t]._to:this[t]})),s}}Es.defaults={},Es.defaultRoutes=void 0;const Rs={values:t=>s(t)?t:""+t,numeric(t,e,i){if(0===t)return"0";const s=this.chart.options.locale;let n,o=t;if(i.length>1){const e=Math.max(Math.abs(i[0].value),Math.abs(i[i.length-1].value));(e<1e-4||e>1e15)&&(n="scientific"),o=function(t,e){let i=e.length>3?e[2].value-e[1].value:e[1].value-e[0].value;Math.abs(i)>=1&&t!==Math.floor(t)&&(i=t-Math.floor(t));return i}(t,i)}const a=I(Math.abs(o)),r=Math.max(Math.min(-1*Math.floor(a),20),0),l={notation:n,minimumFractionDigits:r,maximumFractionDigits:r};return Object.assign(l,this.options.ticks.format),li(t,s,l)},logarithmic(t,e,i){if(0===t)return"0";const s=t/Math.pow(10,Math.floor(I(t)));return 1===s||2===s||5===s?Rs.numeric.call(this,t,e,i):""}};var Is={formatters:Rs};function zs(t,e){const s=t.options.ticks,n=s.maxTicksLimit||function(t){const e=t.options.offset,i=t._tickSize(),s=t._length/i+(e?0:1),n=t._maxLength/i;return Math.floor(Math.min(s,n))}(t),o=s.major.enabled?function(t){const e=[];let i,s;for(i=0,s=t.length;in)return function(t,e,i,s){let n,o=0,a=i[0];for(s=Math.ceil(s),n=0;nn)return e}return Math.max(n,1)}(o,e,n);if(a>0){let t,s;const n=a>1?Math.round((l-r)/(a-1)):null;for(Fs(e,h,c,i(n)?0:r-n,r),t=0,s=a-1;te.lineWidth,tickColor:(t,e)=>e.color,offset:!1,borderDash:[],borderDashOffset:0,borderWidth:1},title:{display:!1,text:"",padding:{top:4,bottom:4}},ticks:{minRotation:0,maxRotation:50,mirror:!1,textStrokeWidth:0,textStrokeColor:"",padding:3,display:!0,autoSkip:!0,autoSkipPadding:3,labelOffset:0,callback:Is.formatters.values,minor:{},major:{},align:"center",crossAlign:"near",showLabelBackdrop:!1,backdropColor:"rgba(255, 255, 255, 0.75)",backdropPadding:2}}),ne.route("scale.ticks","color","","color"),ne.route("scale.grid","color","","borderColor"),ne.route("scale.grid","borderColor","","borderColor"),ne.route("scale.title","color","","color"),ne.describe("scale",{_fallback:!1,_scriptable:t=>!t.startsWith("before")&&!t.startsWith("after")&&"callback"!==t&&"parser"!==t,_indexable:t=>"borderDash"!==t&&"tickBorderDash"!==t}),ne.describe("scales",{_fallback:"scale"}),ne.describe("scale.ticks",{_scriptable:t=>"backdropPadding"!==t&&"callback"!==t,_indexable:t=>"backdropPadding"!==t});const Vs=(t,e,i)=>"top"===e||"left"===e?t[e]+i:t[e]-i;function Bs(t,e){const i=[],s=t.length/e,n=t.length;let o=0;for(;oa+r)))return h}function Ws(t){return t.drawTicks?t.tickLength:0}function js(t,e){if(!t.display)return 0;const i=mi(t.font,e),n=pi(t.padding);return(s(t.text)?t.text.length:1)*i.lineHeight+n.height}function Hs(t,e,i){let s=dt(t);return(i&&"right"!==e||!i&&"right"===e)&&(s=(t=>"left"===t?"right":"right"===t?"left":t)(s)),s}class $s extends Es{constructor(t){super(),this.id=t.id,this.type=t.type,this.options=void 0,this.ctx=t.ctx,this.chart=t.chart,this.top=void 0,this.bottom=void 0,this.left=void 0,this.right=void 0,this.width=void 0,this.height=void 0,this._margins={left:0,right:0,top:0,bottom:0},this.maxWidth=void 0,this.maxHeight=void 0,this.paddingTop=void 0,this.paddingBottom=void 0,this.paddingLeft=void 0,this.paddingRight=void 0,this.axis=void 0,this.labelRotation=void 0,this.min=void 0,this.max=void 0,this._range=void 0,this.ticks=[],this._gridLineItems=null,this._labelItems=null,this._labelSizes=null,this._length=0,this._maxLength=0,this._longestTextCache={},this._startPixel=void 0,this._endPixel=void 0,this._reversePixels=!1,this._userMax=void 0,this._userMin=void 0,this._suggestedMax=void 0,this._suggestedMin=void 0,this._ticksLength=0,this._borderValue=0,this._cache={},this._dataLimitsCached=!1,this.$context=void 0}init(t){this.options=t.setContext(this.getContext()),this.axis=t.axis,this._userMin=this.parse(t.min),this._userMax=this.parse(t.max),this._suggestedMin=this.parse(t.suggestedMin),this._suggestedMax=this.parse(t.suggestedMax)}parse(t,e){return t}getUserBounds(){let{_userMin:t,_userMax:e,_suggestedMin:i,_suggestedMax:s}=this;return t=a(t,Number.POSITIVE_INFINITY),e=a(e,Number.NEGATIVE_INFINITY),i=a(i,Number.POSITIVE_INFINITY),s=a(s,Number.NEGATIVE_INFINITY),{min:a(t,i),max:a(e,s),minDefined:o(t),maxDefined:o(e)}}getMinMax(t){let e,{min:i,max:s,minDefined:n,maxDefined:o}=this.getUserBounds();if(n&&o)return{min:i,max:s};const r=this.getMatchingVisibleMetas();for(let a=0,l=r.length;as?s:i,s=n&&i>s?i:s,{min:a(i,a(s,i)),max:a(s,a(i,s))}}getPadding(){return{left:this.paddingLeft||0,top:this.paddingTop||0,right:this.paddingRight||0,bottom:this.paddingBottom||0}}getTicks(){return this.ticks}getLabels(){const t=this.chart.data;return this.options.labels||(this.isHorizontal()?t.xLabels:t.yLabels)||t.labels||[]}beforeLayout(){this._cache={},this._dataLimitsCached=!1}beforeUpdate(){c(this.options.beforeUpdate,[this])}update(t,e,i){const{beginAtZero:s,grace:n,ticks:o}=this.options,a=o.sampleSize;this.beforeUpdate(),this.maxWidth=t,this.maxHeight=e,this._margins=i=Object.assign({left:0,right:0,top:0,bottom:0},i),this.ticks=null,this._labelSizes=null,this._gridLineItems=null,this._labelItems=null,this.beforeSetDimensions(),this.setDimensions(),this.afterSetDimensions(),this._maxLength=this.isHorizontal()?this.width+i.left+i.right:this.height+i.top+i.bottom,this._dataLimitsCached||(this.beforeDataLimits(),this.determineDataLimits(),this.afterDataLimits(),this._range=xi(this,n,s),this._dataLimitsCached=!0),this.beforeBuildTicks(),this.ticks=this.buildTicks()||[],this.afterBuildTicks();const r=a=n||i<=1||!this.isHorizontal())return void(this.labelRotation=s);const h=this._getLabelSizes(),c=h.widest.width,d=h.highest.height,u=Z(this.chart.width-c,0,this.maxWidth);o=t.offset?this.maxWidth/i:u/(i-1),c+6>o&&(o=u/(i-(t.offset?.5:1)),a=this.maxHeight-Ws(t.grid)-e.padding-js(t.title,this.chart.options.font),r=Math.sqrt(c*c+d*d),l=$(Math.min(Math.asin(Z((h.highest.height+6)/o,-1,1)),Math.asin(Z(a/r,-1,1))-Math.asin(Z(d/r,-1,1)))),l=Math.max(s,Math.min(n,l))),this.labelRotation=l}afterCalculateLabelRotation(){c(this.options.afterCalculateLabelRotation,[this])}afterAutoSkip(){}beforeFit(){c(this.options.beforeFit,[this])}fit(){const t={width:0,height:0},{chart:e,options:{ticks:i,title:s,grid:n}}=this,o=this._isVisible(),a=this.isHorizontal();if(o){const o=js(s,e.options.font);if(a?(t.width=this.maxWidth,t.height=Ws(n)+o):(t.height=this.maxHeight,t.width=Ws(n)+o),i.display&&this.ticks.length){const{first:e,last:s,widest:n,highest:o}=this._getLabelSizes(),r=2*i.padding,l=H(this.labelRotation),h=Math.cos(l),c=Math.sin(l);if(a){const e=i.mirror?0:c*n.width+h*o.height;t.height=Math.min(this.maxHeight,t.height+e+r)}else{const e=i.mirror?0:h*n.width+c*o.height;t.width=Math.min(this.maxWidth,t.width+e+r)}this._calculatePadding(e,s,c,h)}}this._handleMargins(),a?(this.width=this._length=e.width-this._margins.left-this._margins.right,this.height=t.height):(this.width=t.width,this.height=this._length=e.height-this._margins.top-this._margins.bottom)}_calculatePadding(t,e,i,s){const{ticks:{align:n,padding:o},position:a}=this.options,r=0!==this.labelRotation,l="top"!==a&&"x"===this.axis;if(this.isHorizontal()){const a=this.getPixelForTick(0)-this.left,h=this.right-this.getPixelForTick(this.ticks.length-1);let c=0,d=0;r?l?(c=s*t.width,d=i*e.height):(c=i*t.height,d=s*e.width):"start"===n?d=e.width:"end"===n?c=t.width:"inner"!==n&&(c=t.width/2,d=e.width/2),this.paddingLeft=Math.max((c-a+o)*this.width/(this.width-a),0),this.paddingRight=Math.max((d-h+o)*this.width/(this.width-h),0)}else{let i=e.height/2,s=t.height/2;"start"===n?(i=0,s=t.height):"end"===n&&(i=e.height,s=0),this.paddingTop=i+o,this.paddingBottom=s+o}}_handleMargins(){this._margins&&(this._margins.left=Math.max(this.paddingLeft,this._margins.left),this._margins.top=Math.max(this.paddingTop,this._margins.top),this._margins.right=Math.max(this.paddingRight,this._margins.right),this._margins.bottom=Math.max(this.paddingBottom,this._margins.bottom))}afterFit(){c(this.options.afterFit,[this])}isHorizontal(){const{axis:t,position:e}=this.options;return"top"===e||"bottom"===e||"x"===t}isFullSize(){return this.options.fullSize}_convertTicksToLabels(t){let e,s;for(this.beforeTickToLabelConversion(),this.generateTickLabels(t),e=0,s=t.length;e{const i=t.gc,s=i.length/2;let n;if(s>e){for(n=0;n({width:a[t]||0,height:r[t]||0});return{first:k(0),last:k(e-1),widest:k(w),highest:k(M),widths:a,heights:r}}getLabelForValue(t){return t}getPixelForValue(t,e){return NaN}getValueForPixel(t){}getPixelForTick(t){const e=this.ticks;return t<0||t>e.length-1?null:this.getPixelForValue(e[t].value)}getPixelForDecimal(t){this._reversePixels&&(t=1-t);const e=this._startPixel+t*this._length;return J(this._alignToPixels?ve(this.chart,e,0):e)}getDecimalForPixel(t){const e=(t-this._startPixel)/this._length;return this._reversePixels?1-e:e}getBasePixel(){return this.getPixelForValue(this.getBaseValue())}getBaseValue(){const{min:t,max:e}=this;return t<0&&e<0?e:t>0&&e>0?t:0}getContext(t){const e=this.ticks||[];if(t>=0&&ta*s?a/i:r/s:r*s0}_computeGridLineItems(t){const e=this.axis,i=this.chart,s=this.options,{grid:o,position:a}=s,l=o.offset,h=this.isHorizontal(),c=this.ticks.length+(l?1:0),d=Ws(o),u=[],f=o.setContext(this.getContext()),g=f.drawBorder?f.borderWidth:0,p=g/2,m=function(t){return ve(i,t,g)};let b,x,_,y,v,w,M,k,S,P,D,O;if("top"===a)b=m(this.bottom),w=this.bottom-d,k=b-p,P=m(t.top)+p,O=t.bottom;else if("bottom"===a)b=m(this.top),P=t.top,O=m(t.bottom)-p,w=b+p,k=this.top+d;else if("left"===a)b=m(this.right),v=this.right-d,M=b-p,S=m(t.left)+p,D=t.right;else if("right"===a)b=m(this.left),S=t.left,D=m(t.right)-p,v=b+p,M=this.left+d;else if("x"===e){if("center"===a)b=m((t.top+t.bottom)/2+.5);else if(n(a)){const t=Object.keys(a)[0],e=a[t];b=m(this.chart.scales[t].getPixelForValue(e))}P=t.top,O=t.bottom,w=b+p,k=w+d}else if("y"===e){if("center"===a)b=m((t.left+t.right)/2);else if(n(a)){const t=Object.keys(a)[0],e=a[t];b=m(this.chart.scales[t].getPixelForValue(e))}v=b-p,M=v-d,S=t.left,D=t.right}const C=r(s.ticks.maxTicksLimit,c),A=Math.max(1,Math.ceil(c/C));for(x=0;xe.value===t));if(i>=0){return e.setContext(this.getContext(i)).lineWidth}return 0}drawGrid(t){const e=this.options.grid,i=this.ctx,s=this._gridLineItems||(this._gridLineItems=this._computeGridLineItems(t));let n,o;const a=(t,e,s)=>{s.width&&s.color&&(i.save(),i.lineWidth=s.width,i.strokeStyle=s.color,i.setLineDash(s.borderDash||[]),i.lineDashOffset=s.borderDashOffset,i.beginPath(),i.moveTo(t.x,t.y),i.lineTo(e.x,e.y),i.stroke(),i.restore())};if(e.display)for(n=0,o=s.length;n{this.drawBackground(),this.drawGrid(t),this.drawTitle()}},{z:i+1,draw:()=>{this.drawBorder()}},{z:e,draw:t=>{this.drawLabels(t)}}]:[{z:e,draw:t=>{this.draw(t)}}]}getMatchingVisibleMetas(t){const e=this.chart.getSortedVisibleDatasetMetas(),i=this.axis+"AxisID",s=[];let n,o;for(n=0,o=e.length;n{const s=i.split("."),n=s.pop(),o=[t].concat(s).join("."),a=e[i].split("."),r=a.pop(),l=a.join(".");ne.route(o,n,l,r)}))}(e,t.defaultRoutes);t.descriptors&&ne.describe(e,t.descriptors)}(t,o,i),this.override&&ne.override(t.id,t.overrides)),o}get(t){return this.items[t]}unregister(t){const e=this.items,i=t.id,s=this.scope;i in e&&delete e[i],s&&i in ne[s]&&(delete ne[s][i],this.override&&delete te[i])}}var Us=new class{constructor(){this.controllers=new Ys(Ls,"datasets",!0),this.elements=new Ys(Es,"elements"),this.plugins=new Ys(Object,"plugins"),this.scales=new Ys($s,"scales"),this._typedRegistries=[this.controllers,this.scales,this.elements]}add(...t){this._each("register",t)}remove(...t){this._each("unregister",t)}addControllers(...t){this._each("register",t,this.controllers)}addElements(...t){this._each("register",t,this.elements)}addPlugins(...t){this._each("register",t,this.plugins)}addScales(...t){this._each("register",t,this.scales)}getController(t){return this._get(t,this.controllers,"controller")}getElement(t){return this._get(t,this.elements,"element")}getPlugin(t){return this._get(t,this.plugins,"plugin")}getScale(t){return this._get(t,this.scales,"scale")}removeControllers(...t){this._each("unregister",t,this.controllers)}removeElements(...t){this._each("unregister",t,this.elements)}removePlugins(...t){this._each("unregister",t,this.plugins)}removeScales(...t){this._each("unregister",t,this.scales)}_each(t,e,i){[...e].forEach((e=>{const s=i||this._getRegistryForType(e);i||s.isForType(e)||s===this.plugins&&e.id?this._exec(t,s,e):d(e,(e=>{const s=i||this._getRegistryForType(e);this._exec(t,s,e)}))}))}_exec(t,e,i){const s=w(t);c(i["before"+s],[],i),e[t](i),c(i["after"+s],[],i)}_getRegistryForType(t){for(let e=0;et.filter((t=>!e.some((e=>t.plugin.id===e.plugin.id))));this._notify(s(e,i),t,"stop"),this._notify(s(i,e),t,"start")}}function qs(t,e){return e||!1!==t?!0===t?{}:t:null}function Ks(t,{plugin:e,local:i},s,n){const o=t.pluginScopeKeys(e),a=t.getOptionScopes(s,o);return i&&e.defaults&&a.push(e.defaults),t.createResolver(a,n,[""],{scriptable:!1,indexable:!1,allKeys:!0})}function Gs(t,e){const i=ne.datasets[t]||{};return((e.datasets||{})[t]||{}).indexAxis||e.indexAxis||i.indexAxis||"x"}function Zs(t,e){return"x"===t||"y"===t?t:e.axis||("top"===(i=e.position)||"bottom"===i?"x":"left"===i||"right"===i?"y":void 0)||t.charAt(0).toLowerCase();var i}function Js(t){const e=t.options||(t.options={});e.plugins=r(e.plugins,{}),e.scales=function(t,e){const i=te[t.type]||{scales:{}},s=e.scales||{},o=Gs(t.type,e),a=Object.create(null),r=Object.create(null);return Object.keys(s).forEach((t=>{const e=s[t];if(!n(e))return console.error(`Invalid scale configuration for scale: ${t}`);if(e._proxy)return console.warn(`Ignoring resolver passed as options for scale: ${t}`);const l=Zs(t,e),h=function(t,e){return t===e?"_index_":"_value_"}(l,o),c=i.scales||{};a[l]=a[l]||t,r[t]=b(Object.create(null),[{axis:l},e,c[l],c[h]])})),t.data.datasets.forEach((i=>{const n=i.type||t.type,o=i.indexAxis||Gs(n,e),l=(te[n]||{}).scales||{};Object.keys(l).forEach((t=>{const e=function(t,e){let i=t;return"_index_"===t?i=e:"_value_"===t&&(i="x"===e?"y":"x"),i}(t,o),n=i[e+"AxisID"]||a[e]||e;r[n]=r[n]||Object.create(null),b(r[n],[{axis:e},s[n],l[t]])}))})),Object.keys(r).forEach((t=>{const e=r[t];b(e,[ne.scales[e.type],ne.scale])})),r}(t,e)}function Qs(t){return(t=t||{}).datasets=t.datasets||[],t.labels=t.labels||[],t}const tn=new Map,en=new Set;function sn(t,e){let i=tn.get(t);return i||(i=e(),tn.set(t,i),en.add(i)),i}const nn=(t,e,i)=>{const s=y(e,i);void 0!==s&&t.add(s)};class on{constructor(t){this._config=function(t){return(t=t||{}).data=Qs(t.data),Js(t),t}(t),this._scopeCache=new Map,this._resolverCache=new Map}get platform(){return this._config.platform}get type(){return this._config.type}set type(t){this._config.type=t}get data(){return this._config.data}set data(t){this._config.data=Qs(t)}get options(){return this._config.options}set options(t){this._config.options=t}get plugins(){return this._config.plugins}update(){const t=this._config;this.clearCache(),Js(t)}clearCache(){this._scopeCache.clear(),this._resolverCache.clear()}datasetScopeKeys(t){return sn(t,(()=>[[`datasets.${t}`,""]]))}datasetAnimationScopeKeys(t,e){return sn(`${t}.transition.${e}`,(()=>[[`datasets.${t}.transitions.${e}`,`transitions.${e}`],[`datasets.${t}`,""]]))}datasetElementScopeKeys(t,e){return sn(`${t}-${e}`,(()=>[[`datasets.${t}.elements.${e}`,`datasets.${t}`,`elements.${e}`,""]]))}pluginScopeKeys(t){const e=t.id;return sn(`${this.type}-plugin-${e}`,(()=>[[`plugins.${e}`,...t.additionalOptionScopes||[]]]))}_cachedScopes(t,e){const i=this._scopeCache;let s=i.get(t);return s&&!e||(s=new Map,i.set(t,s)),s}getOptionScopes(t,e,i){const{options:s,type:n}=this,o=this._cachedScopes(t,i),a=o.get(e);if(a)return a;const r=new Set;e.forEach((e=>{t&&(r.add(t),e.forEach((e=>nn(r,t,e)))),e.forEach((t=>nn(r,s,t))),e.forEach((t=>nn(r,te[n]||{},t))),e.forEach((t=>nn(r,ne,t))),e.forEach((t=>nn(r,ee,t)))}));const l=Array.from(r);return 0===l.length&&l.push(Object.create(null)),en.has(e)&&o.set(e,l),l}chartOptionScopes(){const{options:t,type:e}=this;return[t,te[e]||{},ne.datasets[e]||{},{type:e},ne,ee]}resolveNamedOptions(t,e,i,n=[""]){const o={$shared:!0},{resolver:a,subPrefixes:r}=an(this._resolverCache,t,n);let l=a;if(function(t,e){const{isScriptable:i,isIndexable:n}=Ie(t);for(const o of e){const e=i(o),a=n(o),r=(a||e)&&t[o];if(e&&(k(r)||rn(r))||a&&s(r))return!0}return!1}(a,e)){o.$shared=!1;l=Re(a,i=k(i)?i():i,this.createResolver(t,i,r))}for(const t of e)o[t]=l[t];return o}createResolver(t,e,i=[""],s){const{resolver:o}=an(this._resolverCache,t,i);return n(e)?Re(o,e,void 0,s):o}}function an(t,e,i){let s=t.get(e);s||(s=new Map,t.set(e,s));const n=i.join();let o=s.get(n);if(!o){o={resolver:Ee(e,i),subPrefixes:i.filter((t=>!t.toLowerCase().includes("hover")))},s.set(n,o)}return o}const rn=t=>n(t)&&Object.getOwnPropertyNames(t).reduce(((e,i)=>e||k(t[i])),!1);const ln=["top","bottom","left","right","chartArea"];function hn(t,e){return"top"===t||"bottom"===t||-1===ln.indexOf(t)&&"x"===e}function cn(t,e){return function(i,s){return i[t]===s[t]?i[e]-s[e]:i[t]-s[t]}}function dn(t){const e=t.chart,i=e.options.animation;e.notifyPlugins("afterRender"),c(i&&i.onComplete,[t],e)}function un(t){const e=t.chart,i=e.options.animation;c(i&&i.onProgress,[t],e)}function fn(t){return oe()&&"string"==typeof t?t=document.getElementById(t):t&&t.length&&(t=t[0]),t&&t.canvas&&(t=t.canvas),t}const gn={},pn=t=>{const e=fn(t);return Object.values(gn).filter((t=>t.canvas===e)).pop()};function mn(t,e,i){const s=Object.keys(t);for(const n of s){const s=+n;if(s>=e){const o=t[n];delete t[n],(i>0||s>e)&&(t[s+i]=o)}}}class bn{constructor(t,i){const s=this.config=new on(i),n=fn(t),o=pn(n);if(o)throw new Error("Canvas is already in use. Chart with ID '"+o.id+"' must be destroyed before the canvas with ID '"+o.canvas.id+"' can be reused.");const a=s.createResolver(s.chartOptionScopes(),this.getContext());this.platform=new(s.platform||gs(n)),this.platform.updateConfig(s);const r=this.platform.acquireContext(n,a.aspectRatio),l=r&&r.canvas,h=l&&l.height,c=l&&l.width;this.id=e(),this.ctx=r,this.canvas=l,this.width=c,this.height=h,this._options=a,this._aspectRatio=this.aspectRatio,this._layers=[],this._metasets=[],this._stacks=void 0,this.boxes=[],this.currentDevicePixelRatio=void 0,this.chartArea=void 0,this._active=[],this._lastEvent=void 0,this._listeners={},this._responsiveListeners=void 0,this._sortedMetasets=[],this.scales={},this._plugins=new Xs,this.$proxies={},this._hiddenIndices={},this.attached=!1,this._animationsDisabled=void 0,this.$context=void 0,this._doResize=ct((t=>this.update(t)),a.resizeDelay||0),this._dataChanges=[],gn[this.id]=this,r&&l?(mt.listen(this,"complete",dn),mt.listen(this,"progress",un),this._initialize(),this.attached&&this.update()):console.error("Failed to create chart: can't acquire context from the given item")}get aspectRatio(){const{options:{aspectRatio:t,maintainAspectRatio:e},width:s,height:n,_aspectRatio:o}=this;return i(t)?e&&o?o:n?s/n:null:t}get data(){return this.config.data}set data(t){this.config.data=t}get options(){return this._options}set options(t){this.config.options=t}_initialize(){return this.notifyPlugins("beforeInit"),this.options.responsive?this.resize():pe(this,this.options.devicePixelRatio),this.bindEvents(),this.notifyPlugins("afterInit"),this}clear(){return we(this.canvas,this.ctx),this}stop(){return mt.stop(this),this}resize(t,e){mt.running(this)?this._resizeBeforeDraw={width:t,height:e}:this._resize(t,e)}_resize(t,e){const i=this.options,s=this.canvas,n=i.maintainAspectRatio&&this.aspectRatio,o=this.platform.getMaximumSize(s,t,e,n),a=i.devicePixelRatio||this.platform.getDevicePixelRatio(),r=this.width?"resize":"attach";this.width=o.width,this.height=o.height,this._aspectRatio=this.aspectRatio,pe(this,a,!0)&&(this.notifyPlugins("resize",{size:o}),c(i.onResize,[this,o],this),this.attached&&this._doResize(r)&&this.render())}ensureScalesHaveIDs(){d(this.options.scales||{},((t,e)=>{t.id=e}))}buildOrUpdateScales(){const t=this.options,e=t.scales,i=this.scales,s=Object.keys(i).reduce(((t,e)=>(t[e]=!1,t)),{});let n=[];e&&(n=n.concat(Object.keys(e).map((t=>{const i=e[t],s=Zs(t,i),n="r"===s,o="x"===s;return{options:i,dposition:n?"chartArea":o?"bottom":"left",dtype:n?"radialLinear":o?"category":"linear"}})))),d(n,(e=>{const n=e.options,o=n.id,a=Zs(o,n),l=r(n.type,e.dtype);void 0!==n.position&&hn(n.position,a)===hn(e.dposition)||(n.position=e.dposition),s[o]=!0;let h=null;if(o in i&&i[o].type===l)h=i[o];else{h=new(Us.getScale(l))({id:o,type:l,ctx:this.ctx,chart:this}),i[h.id]=h}h.init(n,t)})),d(s,((t,e)=>{t||delete i[e]})),d(i,(t=>{Zi.configure(this,t,t.options),Zi.addBox(this,t)}))}_updateMetasets(){const t=this._metasets,e=this.data.datasets.length,i=t.length;if(t.sort(((t,e)=>t.index-e.index)),i>e){for(let t=e;te.length&&delete this._stacks,t.forEach(((t,i)=>{0===e.filter((e=>e===t._dataset)).length&&this._destroyDatasetMeta(i)}))}buildOrUpdateControllers(){const t=[],e=this.data.datasets;let i,s;for(this._removeUnreferencedMetasets(),i=0,s=e.length;i{this.getDatasetMeta(e).controller.reset()}),this)}reset(){this._resetElements(),this.notifyPlugins("reset")}update(t){const e=this.config;e.update();const i=this._options=e.createResolver(e.chartOptionScopes(),this.getContext()),s=this._animationsDisabled=!i.animation;if(this._updateScales(),this._checkEventBindings(),this._updateHiddenIndices(),this._plugins.invalidate(),!1===this.notifyPlugins("beforeUpdate",{mode:t,cancelable:!0}))return;const n=this.buildOrUpdateControllers();this.notifyPlugins("beforeElementsUpdate");let o=0;for(let t=0,e=this.data.datasets.length;t{t.reset()})),this._updateDatasets(t),this.notifyPlugins("afterUpdate",{mode:t}),this._layers.sort(cn("z","_idx"));const{_active:a,_lastEvent:r}=this;r?this._eventHandler(r,!0):a.length&&this._updateHoverStyles(a,a,!0),this.render()}_updateScales(){d(this.scales,(t=>{Zi.removeBox(this,t)})),this.ensureScalesHaveIDs(),this.buildOrUpdateScales()}_checkEventBindings(){const t=this.options,e=new Set(Object.keys(this._listeners)),i=new Set(t.events);S(e,i)&&!!this._responsiveListeners===t.responsive||(this.unbindEvents(),this.bindEvents())}_updateHiddenIndices(){const{_hiddenIndices:t}=this,e=this._getUniformDataChanges()||[];for(const{method:i,start:s,count:n}of e){mn(t,s,"_removeElements"===i?-n:n)}}_getUniformDataChanges(){const t=this._dataChanges;if(!t||!t.length)return;this._dataChanges=[];const e=this.data.datasets.length,i=e=>new Set(t.filter((t=>t[0]===e)).map(((t,e)=>e+","+t.splice(1).join(",")))),s=i(0);for(let t=1;tt.split(","))).map((t=>({method:t[1],start:+t[2],count:+t[3]})))}_updateLayout(t){if(!1===this.notifyPlugins("beforeLayout",{cancelable:!0}))return;Zi.update(this,this.width,this.height,t);const e=this.chartArea,i=e.width<=0||e.height<=0;this._layers=[],d(this.boxes,(t=>{i&&"chartArea"===t.position||(t.configure&&t.configure(),this._layers.push(...t._layers()))}),this),this._layers.forEach(((t,e)=>{t._idx=e})),this.notifyPlugins("afterLayout")}_updateDatasets(t){if(!1!==this.notifyPlugins("beforeDatasetsUpdate",{mode:t,cancelable:!0})){for(let t=0,e=this.data.datasets.length;t=0;--e)this._drawDataset(t[e]);this.notifyPlugins("afterDatasetsDraw")}_drawDataset(t){const e=this.ctx,i=t._clip,s=!i.disabled,n=this.chartArea,o={meta:t,index:t.index,cancelable:!0};!1!==this.notifyPlugins("beforeDatasetDraw",o)&&(s&&Pe(e,{left:!1===i.left?0:n.left-i.left,right:!1===i.right?this.width:n.right+i.right,top:!1===i.top?0:n.top-i.top,bottom:!1===i.bottom?this.height:n.bottom+i.bottom}),t.controller.draw(),s&&De(e),o.cancelable=!1,this.notifyPlugins("afterDatasetDraw",o))}isPointInArea(t){return Se(t,this.chartArea,this._minPadding)}getElementsAtEventForMode(t,e,i,s){const n=Vi.modes[e];return"function"==typeof n?n(this,t,i,s):[]}getDatasetMeta(t){const e=this.data.datasets[t],i=this._metasets;let s=i.filter((t=>t&&t._dataset===e)).pop();return s||(s={type:null,data:[],dataset:null,controller:null,hidden:null,xAxisID:null,yAxisID:null,order:e&&e.order||0,index:t,_dataset:e,_parsed:[],_sorted:!1},i.push(s)),s}getContext(){return this.$context||(this.$context=_i(null,{chart:this,type:"chart"}))}getVisibleDatasetCount(){return this.getSortedVisibleDatasetMetas().length}isDatasetVisible(t){const e=this.data.datasets[t];if(!e)return!1;const i=this.getDatasetMeta(t);return"boolean"==typeof i.hidden?!i.hidden:!e.hidden}setDatasetVisibility(t,e){this.getDatasetMeta(t).hidden=!e}toggleDataVisibility(t){this._hiddenIndices[t]=!this._hiddenIndices[t]}getDataVisibility(t){return!this._hiddenIndices[t]}_updateVisibility(t,e,i){const s=i?"show":"hide",n=this.getDatasetMeta(t),o=n.controller._resolveAnimations(void 0,s);M(e)?(n.data[e].hidden=!i,this.update()):(this.setDatasetVisibility(t,i),o.update(n,{visible:i}),this.update((e=>e.datasetIndex===t?s:void 0)))}hide(t,e){this._updateVisibility(t,e,!1)}show(t,e){this._updateVisibility(t,e,!0)}_destroyDatasetMeta(t){const e=this._metasets[t];e&&e.controller&&e.controller._destroy(),delete this._metasets[t]}_stop(){let t,e;for(this.stop(),mt.remove(this),t=0,e=this.data.datasets.length;t{e.addEventListener(this,i,s),t[i]=s},s=(t,e,i)=>{t.offsetX=e,t.offsetY=i,this._eventHandler(t)};d(this.options.events,(t=>i(t,s)))}bindResponsiveEvents(){this._responsiveListeners||(this._responsiveListeners={});const t=this._responsiveListeners,e=this.platform,i=(i,s)=>{e.addEventListener(this,i,s),t[i]=s},s=(i,s)=>{t[i]&&(e.removeEventListener(this,i,s),delete t[i])},n=(t,e)=>{this.canvas&&this.resize(t,e)};let o;const a=()=>{s("attach",a),this.attached=!0,this.resize(),i("resize",n),i("detach",o)};o=()=>{this.attached=!1,s("resize",n),this._stop(),this._resize(0,0),i("attach",a)},e.isAttached(this.canvas)?a():o()}unbindEvents(){d(this._listeners,((t,e)=>{this.platform.removeEventListener(this,e,t)})),this._listeners={},d(this._responsiveListeners,((t,e)=>{this.platform.removeEventListener(this,e,t)})),this._responsiveListeners=void 0}updateHoverStyle(t,e,i){const s=i?"set":"remove";let n,o,a,r;for("dataset"===e&&(n=this.getDatasetMeta(t[0].datasetIndex),n.controller["_"+s+"DatasetHoverStyle"]()),a=0,r=t.length;a{const i=this.getDatasetMeta(t);if(!i)throw new Error("No dataset found at index "+t);return{datasetIndex:t,element:i.data[e],index:e}}));!u(i,e)&&(this._active=i,this._lastEvent=null,this._updateHoverStyles(i,e))}notifyPlugins(t,e,i){return this._plugins.notify(this,t,e,i)}_updateHoverStyles(t,e,i){const s=this.options.hover,n=(t,e)=>t.filter((t=>!e.some((e=>t.datasetIndex===e.datasetIndex&&t.index===e.index)))),o=n(e,t),a=i?t:n(t,e);o.length&&this.updateHoverStyle(o,s.mode,!1),a.length&&s.mode&&this.updateHoverStyle(a,s.mode,!0)}_eventHandler(t,e){const i={event:t,replay:e,cancelable:!0,inChartArea:this.isPointInArea(t)},s=e=>(e.options.events||this.options.events).includes(t.native.type);if(!1===this.notifyPlugins("beforeEvent",i,s))return;const n=this._handleEvent(t,e,i.inChartArea);return i.cancelable=!1,this.notifyPlugins("afterEvent",i,s),(n||i.changed)&&this.render(),this}_handleEvent(t,e,i){const{_active:s=[],options:n}=this,o=e,a=this._getActiveElements(t,s,i,o),r=P(t),l=function(t,e,i,s){return i&&"mouseout"!==t.type?s?e:t:null}(t,this._lastEvent,i,r);i&&(this._lastEvent=null,c(n.onHover,[t,a,this],this),r&&c(n.onClick,[t,a,this],this));const h=!u(a,s);return(h||e)&&(this._active=a,this._updateHoverStyles(a,s,e)),this._lastEvent=l,h}_getActiveElements(t,e,i,s){if("mouseout"===t.type)return[];if(!i)return e;const n=this.options.hover;return this.getElementsAtEventForMode(t,n.mode,n,s)}}const xn=()=>d(bn.instances,(t=>t._plugins.invalidate())),_n=!0;function yn(){throw new Error("This method is not implemented: Check that a complete date adapter is provided.")}Object.defineProperties(bn,{defaults:{enumerable:_n,value:ne},instances:{enumerable:_n,value:gn},overrides:{enumerable:_n,value:te},registry:{enumerable:_n,value:Us},version:{enumerable:_n,value:"3.9.1"},getChart:{enumerable:_n,value:pn},register:{enumerable:_n,value:(...t)=>{Us.add(...t),xn()}},unregister:{enumerable:_n,value:(...t)=>{Us.remove(...t),xn()}}});class vn{constructor(t){this.options=t||{}}init(t){}formats(){return yn()}parse(t,e){return yn()}format(t,e){return yn()}add(t,e,i){return yn()}diff(t,e,i){return yn()}startOf(t,e,i){return yn()}endOf(t,e){return yn()}}vn.override=function(t){Object.assign(vn.prototype,t)};var wn={_date:vn};function Mn(t){const e=t.iScale,i=function(t,e){if(!t._cache.$bar){const i=t.getMatchingVisibleMetas(e);let s=[];for(let e=0,n=i.length;et-e)))}return t._cache.$bar}(e,t.type);let s,n,o,a,r=e._length;const l=()=>{32767!==o&&-32768!==o&&(M(a)&&(r=Math.min(r,Math.abs(o-a)||r)),a=o)};for(s=0,n=i.length;sMath.abs(r)&&(l=r,h=a),e[i.axis]=h,e._custom={barStart:l,barEnd:h,start:n,end:o,min:a,max:r}}(t,e,i,n):e[i.axis]=i.parse(t,n),e}function Sn(t,e,i,s){const n=t.iScale,o=t.vScale,a=n.getLabels(),r=n===o,l=[];let h,c,d,u;for(h=i,c=i+s;ht.x,i="left",s="right"):(e=t.baset.controller.options.grouped)),o=s.options.stacked,a=[],r=t=>{const s=t.controller.getParsed(e),n=s&&s[t.vScale.axis];if(i(n)||isNaN(n))return!0};for(const i of n)if((void 0===e||!r(i))&&((!1===o||-1===a.indexOf(i.stack)||void 0===o&&void 0===i.stack)&&a.push(i.stack),i.index===t))break;return a.length||a.push(void 0),a}_getStackCount(t){return this._getStacks(void 0,t).length}_getStackIndex(t,e,i){const s=this._getStacks(t,i),n=void 0!==e?s.indexOf(e):-1;return-1===n?s.length-1:n}_getRuler(){const t=this.options,e=this._cachedMeta,i=e.iScale,s=[];let n,o;for(n=0,o=e.data.length;n=i?1:-1)}(d,e,a)*o,u===a&&(m-=d/2);const t=e.getPixelForDecimal(0),i=e.getPixelForDecimal(1),s=Math.min(t,i),n=Math.max(t,i);m=Math.max(Math.min(m,n),s),c=m+d}if(m===e.getPixelForValue(a)){const t=z(d)*e.getLineWidthForValue(a)/2;m+=t,d-=t}return{size:d,base:m,head:c,center:c+d/2}}_calculateBarIndexPixels(t,e){const s=e.scale,n=this.options,o=n.skipNull,a=r(n.maxBarThickness,1/0);let l,h;if(e.grouped){const s=o?this._getStackCount(t):e.stackCount,r="flex"===n.barThickness?function(t,e,i,s){const n=e.pixels,o=n[t];let a=t>0?n[t-1]:null,r=t=0;--i)e=Math.max(e,t[i].size(this.resolveDataElementOptions(i))/2);return e>0&&e}getLabelAndValue(t){const e=this._cachedMeta,{xScale:i,yScale:s}=e,n=this.getParsed(t),o=i.getLabelForValue(n.x),a=s.getLabelForValue(n.y),r=n._custom;return{label:e.label,value:"("+o+", "+a+(r?", "+r:"")+")"}}update(t){const e=this._cachedMeta.data;this.updateElements(e,0,e.length,t)}updateElements(t,e,i,s){const n="reset"===s,{iScale:o,vScale:a}=this._cachedMeta,{sharedOptions:r,includeOptions:l}=this._getSharedOptions(e,s),h=o.axis,c=a.axis;for(let d=e;d""}}}};class En extends Ls{constructor(t,e){super(t,e),this.enableOptionSharing=!0,this.innerRadius=void 0,this.outerRadius=void 0,this.offsetX=void 0,this.offsetY=void 0}linkScales(){}parse(t,e){const i=this.getDataset().data,s=this._cachedMeta;if(!1===this._parsing)s._parsed=i;else{let o,a,r=t=>+i[t];if(n(i[t])){const{key:t="value"}=this._parsing;r=e=>+y(i[e],t)}for(o=t,a=t+e;oG(t,r,l,!0)?1:Math.max(e,e*i,s,s*i),g=(t,e,s)=>G(t,r,l,!0)?-1:Math.min(e,e*i,s,s*i),p=f(0,h,d),m=f(L,c,u),b=g(D,h,d),x=g(D+L,c,u);s=(p-b)/2,n=(m-x)/2,o=-(p+b)/2,a=-(m+x)/2}return{ratioX:s,ratioY:n,offsetX:o,offsetY:a}}(u,d,r),b=(i.width-o)/f,x=(i.height-o)/g,_=Math.max(Math.min(b,x)/2,0),y=h(this.options.radius,_),v=(y-Math.max(y*r,0))/this._getVisibleDatasetWeightTotal();this.offsetX=p*y,this.offsetY=m*y,s.total=this.calculateTotal(),this.outerRadius=y-v*this._getRingWeightOffset(this.index),this.innerRadius=Math.max(this.outerRadius-v*c,0),this.updateElements(n,0,n.length,t)}_circumference(t,e){const i=this.options,s=this._cachedMeta,n=this._getCircumference();return e&&i.animation.animateRotate||!this.chart.getDataVisibility(t)||null===s._parsed[t]||s.data[t].hidden?0:this.calculateCircumference(s._parsed[t]*n/O)}updateElements(t,e,i,s){const n="reset"===s,o=this.chart,a=o.chartArea,r=o.options.animation,l=(a.left+a.right)/2,h=(a.top+a.bottom)/2,c=n&&r.animateScale,d=c?0:this.innerRadius,u=c?0:this.outerRadius,{sharedOptions:f,includeOptions:g}=this._getSharedOptions(e,s);let p,m=this._getRotation();for(p=0;p0&&!isNaN(t)?O*(Math.abs(t)/e):0}getLabelAndValue(t){const e=this._cachedMeta,i=this.chart,s=i.data.labels||[],n=li(e._parsed[t],i.options.locale);return{label:s[t]||"",value:n}}getMaxBorderWidth(t){let e=0;const i=this.chart;let s,n,o,a,r;if(!t)for(s=0,n=i.data.datasets.length;s"spacing"!==t,_indexable:t=>"spacing"!==t},En.overrides={aspectRatio:1,plugins:{legend:{labels:{generateLabels(t){const e=t.data;if(e.labels.length&&e.datasets.length){const{labels:{pointStyle:i}}=t.legend.options;return e.labels.map(((e,s)=>{const n=t.getDatasetMeta(0).controller.getStyle(s);return{text:e,fillStyle:n.backgroundColor,strokeStyle:n.borderColor,lineWidth:n.borderWidth,pointStyle:i,hidden:!t.getDataVisibility(s),index:s}}))}return[]}},onClick(t,e,i){i.chart.toggleDataVisibility(e.index),i.chart.update()}},tooltip:{callbacks:{title:()=>"",label(t){let e=t.label;const i=": "+t.formattedValue;return s(e)?(e=e.slice(),e[0]+=i):e+=i,e}}}}};class Rn extends Ls{initialize(){this.enableOptionSharing=!0,this.supportsDecimation=!0,super.initialize()}update(t){const e=this._cachedMeta,{dataset:i,data:s=[],_dataset:n}=e,o=this.chart._animationsDisabled;let{start:a,count:r}=gt(e,s,o);this._drawStart=a,this._drawCount=r,pt(e)&&(a=0,r=s.length),i._chart=this.chart,i._datasetIndex=this.index,i._decimated=!!n._decimated,i.points=s;const l=this.resolveDatasetElementOptions(t);this.options.showLine||(l.borderWidth=0),l.segment=this.options.segment,this.updateElement(i,void 0,{animated:!o,options:l},t),this.updateElements(s,a,r,t)}updateElements(t,e,s,n){const o="reset"===n,{iScale:a,vScale:r,_stacked:l,_dataset:h}=this._cachedMeta,{sharedOptions:c,includeOptions:d}=this._getSharedOptions(e,n),u=a.axis,f=r.axis,{spanGaps:g,segment:p}=this.options,m=B(g)?g:Number.POSITIVE_INFINITY,b=this.chart._animationsDisabled||o||"none"===n;let x=e>0&&this.getParsed(e-1);for(let g=e;g0&&Math.abs(s[u]-x[u])>m,p&&(_.parsed=s,_.raw=h.data[g]),d&&(_.options=c||this.resolveDataElementOptions(g,e.active?"active":n)),b||this.updateElement(e,g,_,n),x=s}}getMaxOverflow(){const t=this._cachedMeta,e=t.dataset,i=e.options&&e.options.borderWidth||0,s=t.data||[];if(!s.length)return i;const n=s[0].size(this.resolveDataElementOptions(0)),o=s[s.length-1].size(this.resolveDataElementOptions(s.length-1));return Math.max(i,n,o)/2}draw(){const t=this._cachedMeta;t.dataset.updateControlPoints(this.chart.chartArea,t.iScale.axis),super.draw()}}Rn.id="line",Rn.defaults={datasetElementType:"line",dataElementType:"point",showLine:!0,spanGaps:!1},Rn.overrides={scales:{_index_:{type:"category"},_value_:{type:"linear"}}};class In extends Ls{constructor(t,e){super(t,e),this.innerRadius=void 0,this.outerRadius=void 0}getLabelAndValue(t){const e=this._cachedMeta,i=this.chart,s=i.data.labels||[],n=li(e._parsed[t].r,i.options.locale);return{label:s[t]||"",value:n}}parseObjectData(t,e,i,s){return Ue.bind(this)(t,e,i,s)}update(t){const e=this._cachedMeta.data;this._updateRadius(),this.updateElements(e,0,e.length,t)}getMinMax(){const t=this._cachedMeta,e={min:Number.POSITIVE_INFINITY,max:Number.NEGATIVE_INFINITY};return t.data.forEach(((t,i)=>{const s=this.getParsed(i).r;!isNaN(s)&&this.chart.getDataVisibility(i)&&(se.max&&(e.max=s))})),e}_updateRadius(){const t=this.chart,e=t.chartArea,i=t.options,s=Math.min(e.right-e.left,e.bottom-e.top),n=Math.max(s/2,0),o=(n-Math.max(i.cutoutPercentage?n/100*i.cutoutPercentage:1,0))/t.getVisibleDatasetCount();this.outerRadius=n-o*this.index,this.innerRadius=this.outerRadius-o}updateElements(t,e,i,s){const n="reset"===s,o=this.chart,a=o.options.animation,r=this._cachedMeta.rScale,l=r.xCenter,h=r.yCenter,c=r.getIndexAngle(0)-.5*D;let d,u=c;const f=360/this.countVisibleElements();for(d=0;d{!isNaN(this.getParsed(i).r)&&this.chart.getDataVisibility(i)&&e++})),e}_computeAngle(t,e,i){return this.chart.getDataVisibility(t)?H(this.resolveDataElementOptions(t,e).angle||i):0}}In.id="polarArea",In.defaults={dataElementType:"arc",animation:{animateRotate:!0,animateScale:!0},animations:{numbers:{type:"number",properties:["x","y","startAngle","endAngle","innerRadius","outerRadius"]}},indexAxis:"r",startAngle:0},In.overrides={aspectRatio:1,plugins:{legend:{labels:{generateLabels(t){const e=t.data;if(e.labels.length&&e.datasets.length){const{labels:{pointStyle:i}}=t.legend.options;return e.labels.map(((e,s)=>{const n=t.getDatasetMeta(0).controller.getStyle(s);return{text:e,fillStyle:n.backgroundColor,strokeStyle:n.borderColor,lineWidth:n.borderWidth,pointStyle:i,hidden:!t.getDataVisibility(s),index:s}}))}return[]}},onClick(t,e,i){i.chart.toggleDataVisibility(e.index),i.chart.update()}},tooltip:{callbacks:{title:()=>"",label:t=>t.chart.data.labels[t.dataIndex]+": "+t.formattedValue}}},scales:{r:{type:"radialLinear",angleLines:{display:!1},beginAtZero:!0,grid:{circular:!0},pointLabels:{display:!1},startAngle:0}}};class zn extends En{}zn.id="pie",zn.defaults={cutout:0,rotation:0,circumference:360,radius:"100%"};class Fn extends Ls{getLabelAndValue(t){const e=this._cachedMeta.vScale,i=this.getParsed(t);return{label:e.getLabels()[t],value:""+e.getLabelForValue(i[e.axis])}}parseObjectData(t,e,i,s){return Ue.bind(this)(t,e,i,s)}update(t){const e=this._cachedMeta,i=e.dataset,s=e.data||[],n=e.iScale.getLabels();if(i.points=s,"resize"!==t){const e=this.resolveDatasetElementOptions(t);this.options.showLine||(e.borderWidth=0);const o={_loop:!0,_fullLoop:n.length===s.length,options:e};this.updateElement(i,void 0,o,t)}this.updateElements(s,0,s.length,t)}updateElements(t,e,i,s){const n=this._cachedMeta.rScale,o="reset"===s;for(let a=e;a0&&this.getParsed(e-1);for(let c=e;c0&&Math.abs(s[f]-_[f])>b,m&&(p.parsed=s,p.raw=h.data[c]),u&&(p.options=d||this.resolveDataElementOptions(c,e.active?"active":n)),x||this.updateElement(e,c,p,n),_=s}this.updateSharedOptions(d,n,c)}getMaxOverflow(){const t=this._cachedMeta,e=t.data||[];if(!this.options.showLine){let t=0;for(let i=e.length-1;i>=0;--i)t=Math.max(t,e[i].size(this.resolveDataElementOptions(i))/2);return t>0&&t}const i=t.dataset,s=i.options&&i.options.borderWidth||0;if(!e.length)return s;const n=e[0].size(this.resolveDataElementOptions(0)),o=e[e.length-1].size(this.resolveDataElementOptions(e.length-1));return Math.max(s,n,o)/2}}Vn.id="scatter",Vn.defaults={datasetElementType:!1,dataElementType:"point",showLine:!1,fill:!1},Vn.overrides={interaction:{mode:"point"},plugins:{tooltip:{callbacks:{title:()=>"",label:t=>"("+t.label+", "+t.formattedValue+")"}}},scales:{x:{type:"linear"},y:{type:"linear"}}};var Bn=Object.freeze({__proto__:null,BarController:Tn,BubbleController:Ln,DoughnutController:En,LineController:Rn,PolarAreaController:In,PieController:zn,RadarController:Fn,ScatterController:Vn});function Nn(t,e,i){const{startAngle:s,pixelMargin:n,x:o,y:a,outerRadius:r,innerRadius:l}=e;let h=n/r;t.beginPath(),t.arc(o,a,r,s-h,i+h),l>n?(h=n/l,t.arc(o,a,l,i+h,s-h,!0)):t.arc(o,a,n,i+L,s-L),t.closePath(),t.clip()}function Wn(t,e,i,s){const n=ui(t.options.borderRadius,["outerStart","outerEnd","innerStart","innerEnd"]);const o=(i-e)/2,a=Math.min(o,s*e/2),r=t=>{const e=(i-Math.min(o,t))*s/2;return Z(t,0,Math.min(o,e))};return{outerStart:r(n.outerStart),outerEnd:r(n.outerEnd),innerStart:Z(n.innerStart,0,a),innerEnd:Z(n.innerEnd,0,a)}}function jn(t,e,i,s){return{x:i+t*Math.cos(e),y:s+t*Math.sin(e)}}function Hn(t,e,i,s,n,o){const{x:a,y:r,startAngle:l,pixelMargin:h,innerRadius:c}=e,d=Math.max(e.outerRadius+s+i-h,0),u=c>0?c+s+i+h:0;let f=0;const g=n-l;if(s){const t=((c>0?c-s:0)+(d>0?d-s:0))/2;f=(g-(0!==t?g*t/(t+s):g))/2}const p=(g-Math.max(.001,g*d-i/D)/d)/2,m=l+p+f,b=n-p-f,{outerStart:x,outerEnd:_,innerStart:y,innerEnd:v}=Wn(e,u,d,b-m),w=d-x,M=d-_,k=m+x/w,S=b-_/M,P=u+y,O=u+v,C=m+y/P,A=b-v/O;if(t.beginPath(),o){if(t.arc(a,r,d,k,S),_>0){const e=jn(M,S,a,r);t.arc(e.x,e.y,_,S,b+L)}const e=jn(O,b,a,r);if(t.lineTo(e.x,e.y),v>0){const e=jn(O,A,a,r);t.arc(e.x,e.y,v,b+L,A+Math.PI)}if(t.arc(a,r,u,b-v/u,m+y/u,!0),y>0){const e=jn(P,C,a,r);t.arc(e.x,e.y,y,C+Math.PI,m-L)}const i=jn(w,m,a,r);if(t.lineTo(i.x,i.y),x>0){const e=jn(w,k,a,r);t.arc(e.x,e.y,x,m-L,k)}}else{t.moveTo(a,r);const e=Math.cos(k)*d+a,i=Math.sin(k)*d+r;t.lineTo(e,i);const s=Math.cos(S)*d+a,n=Math.sin(S)*d+r;t.lineTo(s,n)}t.closePath()}function $n(t,e,i,s,n,o){const{options:a}=e,{borderWidth:r,borderJoinStyle:l}=a,h="inner"===a.borderAlign;r&&(h?(t.lineWidth=2*r,t.lineJoin=l||"round"):(t.lineWidth=r,t.lineJoin=l||"bevel"),e.fullCircles&&function(t,e,i){const{x:s,y:n,startAngle:o,pixelMargin:a,fullCircles:r}=e,l=Math.max(e.outerRadius-a,0),h=e.innerRadius+a;let c;for(i&&Nn(t,e,o+O),t.beginPath(),t.arc(s,n,h,o+O,o,!0),c=0;c=O||G(n,a,l),g=Q(o,h+u,c+u);return f&&g}getCenterPoint(t){const{x:e,y:i,startAngle:s,endAngle:n,innerRadius:o,outerRadius:a}=this.getProps(["x","y","startAngle","endAngle","innerRadius","outerRadius","circumference"],t),{offset:r,spacing:l}=this.options,h=(s+n)/2,c=(o+a+l+r)/2;return{x:e+Math.cos(h)*c,y:i+Math.sin(h)*c}}tooltipPosition(t){return this.getCenterPoint(t)}draw(t){const{options:e,circumference:i}=this,s=(e.offset||0)/2,n=(e.spacing||0)/2,o=e.circular;if(this.pixelMargin="inner"===e.borderAlign?.33:0,this.fullCircles=i>O?Math.floor(i/O):0,0===i||this.innerRadius<0||this.outerRadius<0)return;t.save();let a=0;if(s){a=s/2;const e=(this.startAngle+this.endAngle)/2;t.translate(Math.cos(e)*a,Math.sin(e)*a),this.circumference>=D&&(a=s)}t.fillStyle=e.backgroundColor,t.strokeStyle=e.borderColor;const r=function(t,e,i,s,n){const{fullCircles:o,startAngle:a,circumference:r}=e;let l=e.endAngle;if(o){Hn(t,e,i,s,a+O,n);for(let e=0;er&&o>r;return{count:s,start:l,loop:e.loop,ilen:h(a+(h?r-t:t))%o,_=()=>{f!==g&&(t.lineTo(m,g),t.lineTo(m,f),t.lineTo(m,p))};for(l&&(d=n[x(0)],t.moveTo(d.x,d.y)),c=0;c<=r;++c){if(d=n[x(c)],d.skip)continue;const e=d.x,i=d.y,s=0|e;s===u?(ig&&(g=i),m=(b*m+e)/++b):(_(),t.lineTo(e,i),u=s,b=0,f=g=i),p=i}_()}function Zn(t){const e=t.options,i=e.borderDash&&e.borderDash.length;return!(t._decimated||t._loop||e.tension||"monotone"===e.cubicInterpolationMode||e.stepped||i)?Gn:Kn}Yn.id="arc",Yn.defaults={borderAlign:"center",borderColor:"#fff",borderJoinStyle:void 0,borderRadius:0,borderWidth:2,offset:0,spacing:0,angle:void 0,circular:!0},Yn.defaultRoutes={backgroundColor:"backgroundColor"};const Jn="function"==typeof Path2D;function Qn(t,e,i,s){Jn&&!e.options.segment?function(t,e,i,s){let n=e._path;n||(n=e._path=new Path2D,e.path(n,i,s)&&n.closePath()),Un(t,e.options),t.stroke(n)}(t,e,i,s):function(t,e,i,s){const{segments:n,options:o}=e,a=Zn(e);for(const r of n)Un(t,o,r.style),t.beginPath(),a(t,e,r,{start:i,end:i+s-1})&&t.closePath(),t.stroke()}(t,e,i,s)}class to extends Es{constructor(t){super(),this.animated=!0,this.options=void 0,this._chart=void 0,this._loop=void 0,this._fullLoop=void 0,this._path=void 0,this._points=void 0,this._segments=void 0,this._decimated=!1,this._pointsUpdated=!1,this._datasetIndex=void 0,t&&Object.assign(this,t)}updateControlPoints(t,e){const i=this.options;if((i.tension||"monotone"===i.cubicInterpolationMode)&&!i.stepped&&!this._pointsUpdated){const s=i.spanGaps?this._loop:this._fullLoop;Qe(this._points,i,t,s,e),this._pointsUpdated=!0}}set points(t){this._points=t,delete this._segments,delete this._path,this._pointsUpdated=!1}get points(){return this._points}get segments(){return this._segments||(this._segments=Di(this,this.options.segment))}first(){const t=this.segments,e=this.points;return t.length&&e[t[0].start]}last(){const t=this.segments,e=this.points,i=t.length;return i&&e[t[i-1].end]}interpolate(t,e){const i=this.options,s=t[e],n=this.points,o=Pi(this,{property:e,start:s,end:s});if(!o.length)return;const a=[],r=function(t){return t.stepped?oi:t.tension||"monotone"===t.cubicInterpolationMode?ai:ni}(i);let l,h;for(l=0,h=o.length;l"borderDash"!==t&&"fill"!==t};class io extends Es{constructor(t){super(),this.options=void 0,this.parsed=void 0,this.skip=void 0,this.stop=void 0,t&&Object.assign(this,t)}inRange(t,e,i){const s=this.options,{x:n,y:o}=this.getProps(["x","y"],i);return Math.pow(t-n,2)+Math.pow(e-o,2){uo(t)}))}var go={id:"decimation",defaults:{algorithm:"min-max",enabled:!1},beforeElementsUpdate:(t,e,s)=>{if(!s.enabled)return void fo(t);const n=t.width;t.data.datasets.forEach(((e,o)=>{const{_data:a,indexAxis:r}=e,l=t.getDatasetMeta(o),h=a||e.data;if("y"===bi([r,t.options.indexAxis]))return;if(!l.controller.supportsDecimation)return;const c=t.scales[l.xAxisID];if("linear"!==c.type&&"time"!==c.type)return;if(t.options.parsing)return;let{start:d,count:u}=function(t,e){const i=e.length;let s,n=0;const{iScale:o}=t,{min:a,max:r,minDefined:l,maxDefined:h}=o.getUserBounds();return l&&(n=Z(et(e,o.axis,a).lo,0,i-1)),s=h?Z(et(e,o.axis,r).hi+1,n,i)-n:i-n,{start:n,count:s}}(l,h);if(u<=(s.threshold||4*n))return void uo(e);let f;switch(i(a)&&(e._data=h,delete e.data,Object.defineProperty(e,"data",{configurable:!0,enumerable:!0,get:function(){return this._decimated},set:function(t){this._data=t}})),s.algorithm){case"lttb":f=function(t,e,i,s,n){const o=n.samples||s;if(o>=i)return t.slice(e,e+i);const a=[],r=(i-2)/(o-2);let l=0;const h=e+i-1;let c,d,u,f,g,p=e;for(a[l++]=t[p],c=0;cu&&(u=f,d=t[s],g=s);a[l++]=d,p=g}return a[l++]=t[h],a}(h,d,u,n,s);break;case"min-max":f=function(t,e,s,n){let o,a,r,l,h,c,d,u,f,g,p=0,m=0;const b=[],x=e+s-1,_=t[e].x,y=t[x].x-_;for(o=e;og&&(g=l,d=o),p=(m*p+a.x)/++m;else{const s=o-1;if(!i(c)&&!i(d)){const e=Math.min(c,d),i=Math.max(c,d);e!==u&&e!==s&&b.push({...t[e],x:p}),i!==u&&i!==s&&b.push({...t[i],x:p})}o>0&&s!==u&&b.push(t[s]),b.push(a),h=e,m=0,f=g=l,c=d=u=o}}return b}(h,d,u,n);break;default:throw new Error(`Unsupported decimation algorithm '${s.algorithm}'`)}e._decimated=f}))},destroy(t){fo(t)}};function po(t,e,i,s){if(s)return;let n=e[t],o=i[t];return"angle"===t&&(n=K(n),o=K(o)),{property:t,start:n,end:o}}function mo(t,e,i){for(;e>t;e--){const t=i[e];if(!isNaN(t.x)&&!isNaN(t.y))break}return e}function bo(t,e,i,s){return t&&e?s(t[i],e[i]):t?t[i]:e?e[i]:0}function xo(t,e){let i=[],n=!1;return s(t)?(n=!0,i=t):i=function(t,e){const{x:i=null,y:s=null}=t||{},n=e.points,o=[];return e.segments.forEach((({start:t,end:e})=>{e=mo(t,e,n);const a=n[t],r=n[e];null!==s?(o.push({x:a.x,y:s}),o.push({x:r.x,y:s})):null!==i&&(o.push({x:i,y:a.y}),o.push({x:i,y:r.y}))})),o}(t,e),i.length?new to({points:i,options:{tension:0},_loop:n,_fullLoop:n}):null}function _o(t){return t&&!1!==t.fill}function yo(t,e,i){let s=t[e].fill;const n=[e];let a;if(!i)return s;for(;!1!==s&&-1===n.indexOf(s);){if(!o(s))return s;if(a=t[s],!a)return!1;if(a.visible)return s;n.push(s),s=a.fill}return!1}function vo(t,e,i){const s=function(t){const e=t.options,i=e.fill;let s=r(i&&i.target,i);void 0===s&&(s=!!e.backgroundColor);if(!1===s||null===s)return!1;if(!0===s)return"origin";return s}(t);if(n(s))return!isNaN(s.value)&&s;let a=parseFloat(s);return o(a)&&Math.floor(a)===a?function(t,e,i,s){"-"!==t&&"+"!==t||(i=e+i);if(i===e||i<0||i>=s)return!1;return i}(s[0],e,a,i):["origin","start","end","stack","shape"].indexOf(s)>=0&&s}function wo(t,e,i){const s=[];for(let n=0;n=0;--e){const i=n[e].$filler;i&&(i.line.updateControlPoints(o,i.axis),s&&i.fill&&Po(t.ctx,i,o))}},beforeDatasetsDraw(t,e,i){if("beforeDatasetsDraw"!==i.drawTime)return;const s=t.getSortedVisibleDatasetMetas();for(let e=s.length-1;e>=0;--e){const i=s[e].$filler;_o(i)&&Po(t.ctx,i,t.chartArea)}},beforeDatasetDraw(t,e,i){const s=e.meta.$filler;_o(s)&&"beforeDatasetDraw"===i.drawTime&&Po(t.ctx,s,t.chartArea)},defaults:{propagate:!0,drawTime:"beforeDatasetDraw"}};const Lo=(t,e)=>{let{boxHeight:i=e,boxWidth:s=e}=t;return t.usePointStyle&&(i=Math.min(i,e),s=t.pointStyleWidth||Math.min(s,e)),{boxWidth:s,boxHeight:i,itemHeight:Math.max(e,i)}};class Eo extends Es{constructor(t){super(),this._added=!1,this.legendHitBoxes=[],this._hoveredItem=null,this.doughnutMode=!1,this.chart=t.chart,this.options=t.options,this.ctx=t.ctx,this.legendItems=void 0,this.columnSizes=void 0,this.lineWidths=void 0,this.maxHeight=void 0,this.maxWidth=void 0,this.top=void 0,this.bottom=void 0,this.left=void 0,this.right=void 0,this.height=void 0,this.width=void 0,this._margins=void 0,this.position=void 0,this.weight=void 0,this.fullSize=void 0}update(t,e,i){this.maxWidth=t,this.maxHeight=e,this._margins=i,this.setDimensions(),this.buildLabels(),this.fit()}setDimensions(){this.isHorizontal()?(this.width=this.maxWidth,this.left=this._margins.left,this.right=this.width):(this.height=this.maxHeight,this.top=this._margins.top,this.bottom=this.height)}buildLabels(){const t=this.options.labels||{};let e=c(t.generateLabels,[this.chart],this)||[];t.filter&&(e=e.filter((e=>t.filter(e,this.chart.data)))),t.sort&&(e=e.sort(((e,i)=>t.sort(e,i,this.chart.data)))),this.options.reverse&&e.reverse(),this.legendItems=e}fit(){const{options:t,ctx:e}=this;if(!t.display)return void(this.width=this.height=0);const i=t.labels,s=mi(i.font),n=s.size,o=this._computeTitleHeight(),{boxWidth:a,itemHeight:r}=Lo(i,n);let l,h;e.font=s.string,this.isHorizontal()?(l=this.maxWidth,h=this._fitRows(o,n,a,r)+10):(h=this.maxHeight,l=this._fitCols(o,n,a,r)+10),this.width=Math.min(l,t.maxWidth||this.maxWidth),this.height=Math.min(h,t.maxHeight||this.maxHeight)}_fitRows(t,e,i,s){const{ctx:n,maxWidth:o,options:{labels:{padding:a}}}=this,r=this.legendHitBoxes=[],l=this.lineWidths=[0],h=s+a;let c=t;n.textAlign="left",n.textBaseline="middle";let d=-1,u=-h;return this.legendItems.forEach(((t,f)=>{const g=i+e/2+n.measureText(t.text).width;(0===f||l[l.length-1]+g+2*a>o)&&(c+=h,l[l.length-(f>0?0:1)]=0,u+=h,d++),r[f]={left:0,top:u,row:d,width:g,height:s},l[l.length-1]+=g+a})),c}_fitCols(t,e,i,s){const{ctx:n,maxHeight:o,options:{labels:{padding:a}}}=this,r=this.legendHitBoxes=[],l=this.columnSizes=[],h=o-t;let c=a,d=0,u=0,f=0,g=0;return this.legendItems.forEach(((t,o)=>{const p=i+e/2+n.measureText(t.text).width;o>0&&u+s+2*a>h&&(c+=d+a,l.push({width:d,height:u}),f+=d+a,g++,d=u=0),r[o]={left:f,top:u,col:g,width:p,height:s},d=Math.max(d,p),u+=s+a})),c+=d,l.push({width:d,height:u}),c}adjustHitBoxes(){if(!this.options.display)return;const t=this._computeTitleHeight(),{legendHitBoxes:e,options:{align:i,labels:{padding:s},rtl:n}}=this,o=yi(n,this.left,this.width);if(this.isHorizontal()){let n=0,a=ut(i,this.left+s,this.right-this.lineWidths[n]);for(const r of e)n!==r.row&&(n=r.row,a=ut(i,this.left+s,this.right-this.lineWidths[n])),r.top+=this.top+t+s,r.left=o.leftForLtr(o.x(a),r.width),a+=r.width+s}else{let n=0,a=ut(i,this.top+t+s,this.bottom-this.columnSizes[n].height);for(const r of e)r.col!==n&&(n=r.col,a=ut(i,this.top+t+s,this.bottom-this.columnSizes[n].height)),r.top=a,r.left+=this.left+s,r.left=o.leftForLtr(o.x(r.left),r.width),a+=r.height+s}}isHorizontal(){return"top"===this.options.position||"bottom"===this.options.position}draw(){if(this.options.display){const t=this.ctx;Pe(t,this),this._draw(),De(t)}}_draw(){const{options:t,columnSizes:e,lineWidths:i,ctx:s}=this,{align:n,labels:o}=t,a=ne.color,l=yi(t.rtl,this.left,this.width),h=mi(o.font),{color:c,padding:d}=o,u=h.size,f=u/2;let g;this.drawTitle(),s.textAlign=l.textAlign("left"),s.textBaseline="middle",s.lineWidth=.5,s.font=h.string;const{boxWidth:p,boxHeight:m,itemHeight:b}=Lo(o,u),x=this.isHorizontal(),_=this._computeTitleHeight();g=x?{x:ut(n,this.left+d,this.right-i[0]),y:this.top+d+_,line:0}:{x:this.left+d,y:ut(n,this.top+_+d,this.bottom-e[0].height),line:0},vi(this.ctx,t.textDirection);const y=b+d;this.legendItems.forEach(((v,w)=>{s.strokeStyle=v.fontColor||c,s.fillStyle=v.fontColor||c;const M=s.measureText(v.text).width,k=l.textAlign(v.textAlign||(v.textAlign=o.textAlign)),S=p+f+M;let P=g.x,D=g.y;l.setWidth(this.width),x?w>0&&P+S+d>this.right&&(D=g.y+=y,g.line++,P=g.x=ut(n,this.left+d,this.right-i[g.line])):w>0&&D+y>this.bottom&&(P=g.x=P+e[g.line].width+d,g.line++,D=g.y=ut(n,this.top+_+d,this.bottom-e[g.line].height));!function(t,e,i){if(isNaN(p)||p<=0||isNaN(m)||m<0)return;s.save();const n=r(i.lineWidth,1);if(s.fillStyle=r(i.fillStyle,a),s.lineCap=r(i.lineCap,"butt"),s.lineDashOffset=r(i.lineDashOffset,0),s.lineJoin=r(i.lineJoin,"miter"),s.lineWidth=n,s.strokeStyle=r(i.strokeStyle,a),s.setLineDash(r(i.lineDash,[])),o.usePointStyle){const a={radius:m*Math.SQRT2/2,pointStyle:i.pointStyle,rotation:i.rotation,borderWidth:n},r=l.xPlus(t,p/2);ke(s,a,r,e+f,o.pointStyleWidth&&p)}else{const o=e+Math.max((u-m)/2,0),a=l.leftForLtr(t,p),r=gi(i.borderRadius);s.beginPath(),Object.values(r).some((t=>0!==t))?Le(s,{x:a,y:o,w:p,h:m,radius:r}):s.rect(a,o,p,m),s.fill(),0!==n&&s.stroke()}s.restore()}(l.x(P),D,v),P=ft(k,P+p+f,x?P+S:this.right,t.rtl),function(t,e,i){Ae(s,i.text,t,e+b/2,h,{strikethrough:i.hidden,textAlign:l.textAlign(i.textAlign)})}(l.x(P),D,v),x?g.x+=S+d:g.y+=y})),wi(this.ctx,t.textDirection)}drawTitle(){const t=this.options,e=t.title,i=mi(e.font),s=pi(e.padding);if(!e.display)return;const n=yi(t.rtl,this.left,this.width),o=this.ctx,a=e.position,r=i.size/2,l=s.top+r;let h,c=this.left,d=this.width;if(this.isHorizontal())d=Math.max(...this.lineWidths),h=this.top+l,c=ut(t.align,c,this.right-d);else{const e=this.columnSizes.reduce(((t,e)=>Math.max(t,e.height)),0);h=l+ut(t.align,this.top,this.bottom-e-t.labels.padding-this._computeTitleHeight())}const u=ut(a,c,c+d);o.textAlign=n.textAlign(dt(a)),o.textBaseline="middle",o.strokeStyle=e.color,o.fillStyle=e.color,o.font=i.string,Ae(o,e.text,u,h,i)}_computeTitleHeight(){const t=this.options.title,e=mi(t.font),i=pi(t.padding);return t.display?e.lineHeight+i.height:0}_getLegendItemAt(t,e){let i,s,n;if(Q(t,this.left,this.right)&&Q(e,this.top,this.bottom))for(n=this.legendHitBoxes,i=0;it.chart.options.color,boxWidth:40,padding:10,generateLabels(t){const e=t.data.datasets,{labels:{usePointStyle:i,pointStyle:s,textAlign:n,color:o}}=t.legend.options;return t._getSortedDatasetMetas().map((t=>{const a=t.controller.getStyle(i?0:void 0),r=pi(a.borderWidth);return{text:e[t.index].label,fillStyle:a.backgroundColor,fontColor:o,hidden:!t.visible,lineCap:a.borderCapStyle,lineDash:a.borderDash,lineDashOffset:a.borderDashOffset,lineJoin:a.borderJoinStyle,lineWidth:(r.width+r.height)/4,strokeStyle:a.borderColor,pointStyle:s||a.pointStyle,rotation:a.rotation,textAlign:n||a.textAlign,borderRadius:0,datasetIndex:t.index}}),this)}},title:{color:t=>t.chart.options.color,display:!1,position:"center",text:""}},descriptors:{_scriptable:t=>!t.startsWith("on"),labels:{_scriptable:t=>!["generateLabels","filter","sort"].includes(t)}}};class Io extends Es{constructor(t){super(),this.chart=t.chart,this.options=t.options,this.ctx=t.ctx,this._padding=void 0,this.top=void 0,this.bottom=void 0,this.left=void 0,this.right=void 0,this.width=void 0,this.height=void 0,this.position=void 0,this.weight=void 0,this.fullSize=void 0}update(t,e){const i=this.options;if(this.left=0,this.top=0,!i.display)return void(this.width=this.height=this.right=this.bottom=0);this.width=this.right=t,this.height=this.bottom=e;const n=s(i.text)?i.text.length:1;this._padding=pi(i.padding);const o=n*mi(i.font).lineHeight+this._padding.height;this.isHorizontal()?this.height=o:this.width=o}isHorizontal(){const t=this.options.position;return"top"===t||"bottom"===t}_drawArgs(t){const{top:e,left:i,bottom:s,right:n,options:o}=this,a=o.align;let r,l,h,c=0;return this.isHorizontal()?(l=ut(a,i,n),h=e+t,r=n-i):("left"===o.position?(l=i+t,h=ut(a,s,e),c=-.5*D):(l=n-t,h=ut(a,e,s),c=.5*D),r=s-e),{titleX:l,titleY:h,maxWidth:r,rotation:c}}draw(){const t=this.ctx,e=this.options;if(!e.display)return;const i=mi(e.font),s=i.lineHeight/2+this._padding.top,{titleX:n,titleY:o,maxWidth:a,rotation:r}=this._drawArgs(s);Ae(t,e.text,0,0,i,{color:e.color,maxWidth:a,rotation:r,textAlign:dt(e.align),textBaseline:"middle",translation:[n,o]})}}var zo={id:"title",_element:Io,start(t,e,i){!function(t,e){const i=new Io({ctx:t.ctx,options:e,chart:t});Zi.configure(t,i,e),Zi.addBox(t,i),t.titleBlock=i}(t,i)},stop(t){const e=t.titleBlock;Zi.removeBox(t,e),delete t.titleBlock},beforeUpdate(t,e,i){const s=t.titleBlock;Zi.configure(t,s,i),s.options=i},defaults:{align:"center",display:!1,font:{weight:"bold"},fullSize:!0,padding:10,position:"top",text:"",weight:2e3},defaultRoutes:{color:"color"},descriptors:{_scriptable:!0,_indexable:!1}};const Fo=new WeakMap;var Vo={id:"subtitle",start(t,e,i){const s=new Io({ctx:t.ctx,options:i,chart:t});Zi.configure(t,s,i),Zi.addBox(t,s),Fo.set(t,s)},stop(t){Zi.removeBox(t,Fo.get(t)),Fo.delete(t)},beforeUpdate(t,e,i){const s=Fo.get(t);Zi.configure(t,s,i),s.options=i},defaults:{align:"center",display:!1,font:{weight:"normal"},fullSize:!0,padding:0,position:"top",text:"",weight:1500},defaultRoutes:{color:"color"},descriptors:{_scriptable:!0,_indexable:!1}};const Bo={average(t){if(!t.length)return!1;let e,i,s=0,n=0,o=0;for(e=0,i=t.length;e-1?t.split("\n"):t}function jo(t,e){const{element:i,datasetIndex:s,index:n}=e,o=t.getDatasetMeta(s).controller,{label:a,value:r}=o.getLabelAndValue(n);return{chart:t,label:a,parsed:o.getParsed(n),raw:t.data.datasets[s].data[n],formattedValue:r,dataset:o.getDataset(),dataIndex:n,datasetIndex:s,element:i}}function Ho(t,e){const i=t.chart.ctx,{body:s,footer:n,title:o}=t,{boxWidth:a,boxHeight:r}=e,l=mi(e.bodyFont),h=mi(e.titleFont),c=mi(e.footerFont),u=o.length,f=n.length,g=s.length,p=pi(e.padding);let m=p.height,b=0,x=s.reduce(((t,e)=>t+e.before.length+e.lines.length+e.after.length),0);if(x+=t.beforeBody.length+t.afterBody.length,u&&(m+=u*h.lineHeight+(u-1)*e.titleSpacing+e.titleMarginBottom),x){m+=g*(e.displayColors?Math.max(r,l.lineHeight):l.lineHeight)+(x-g)*l.lineHeight+(x-1)*e.bodySpacing}f&&(m+=e.footerMarginTop+f*c.lineHeight+(f-1)*e.footerSpacing);let _=0;const y=function(t){b=Math.max(b,i.measureText(t).width+_)};return i.save(),i.font=h.string,d(t.title,y),i.font=l.string,d(t.beforeBody.concat(t.afterBody),y),_=e.displayColors?a+2+e.boxPadding:0,d(s,(t=>{d(t.before,y),d(t.lines,y),d(t.after,y)})),_=0,i.font=c.string,d(t.footer,y),i.restore(),b+=p.width,{width:b,height:m}}function $o(t,e,i,s){const{x:n,width:o}=i,{width:a,chartArea:{left:r,right:l}}=t;let h="center";return"center"===s?h=n<=(r+l)/2?"left":"right":n<=o/2?h="left":n>=a-o/2&&(h="right"),function(t,e,i,s){const{x:n,width:o}=s,a=i.caretSize+i.caretPadding;return"left"===t&&n+o+a>e.width||"right"===t&&n-o-a<0||void 0}(h,t,e,i)&&(h="center"),h}function Yo(t,e,i){const s=i.yAlign||e.yAlign||function(t,e){const{y:i,height:s}=e;return it.height-s/2?"bottom":"center"}(t,i);return{xAlign:i.xAlign||e.xAlign||$o(t,e,i,s),yAlign:s}}function Uo(t,e,i,s){const{caretSize:n,caretPadding:o,cornerRadius:a}=t,{xAlign:r,yAlign:l}=i,h=n+o,{topLeft:c,topRight:d,bottomLeft:u,bottomRight:f}=gi(a);let g=function(t,e){let{x:i,width:s}=t;return"right"===e?i-=s:"center"===e&&(i-=s/2),i}(e,r);const p=function(t,e,i){let{y:s,height:n}=t;return"top"===e?s+=i:s-="bottom"===e?n+i:n/2,s}(e,l,h);return"center"===l?"left"===r?g+=h:"right"===r&&(g-=h):"left"===r?g-=Math.max(c,u)+n:"right"===r&&(g+=Math.max(d,f)+n),{x:Z(g,0,s.width-e.width),y:Z(p,0,s.height-e.height)}}function Xo(t,e,i){const s=pi(i.padding);return"center"===e?t.x+t.width/2:"right"===e?t.x+t.width-s.right:t.x+s.left}function qo(t){return No([],Wo(t))}function Ko(t,e){const i=e&&e.dataset&&e.dataset.tooltip&&e.dataset.tooltip.callbacks;return i?t.override(i):t}class Go extends Es{constructor(t){super(),this.opacity=0,this._active=[],this._eventPosition=void 0,this._size=void 0,this._cachedAnimations=void 0,this._tooltipItems=[],this.$animations=void 0,this.$context=void 0,this.chart=t.chart||t._chart,this._chart=this.chart,this.options=t.options,this.dataPoints=void 0,this.title=void 0,this.beforeBody=void 0,this.body=void 0,this.afterBody=void 0,this.footer=void 0,this.xAlign=void 0,this.yAlign=void 0,this.x=void 0,this.y=void 0,this.height=void 0,this.width=void 0,this.caretX=void 0,this.caretY=void 0,this.labelColors=void 0,this.labelPointStyles=void 0,this.labelTextColors=void 0}initialize(t){this.options=t,this._cachedAnimations=void 0,this.$context=void 0}_resolveAnimations(){const t=this._cachedAnimations;if(t)return t;const e=this.chart,i=this.options.setContext(this.getContext()),s=i.enabled&&e.options.animation&&i.animations,n=new ys(this.chart,s);return s._cacheable&&(this._cachedAnimations=Object.freeze(n)),n}getContext(){return this.$context||(this.$context=(t=this.chart.getContext(),e=this,i=this._tooltipItems,_i(t,{tooltip:e,tooltipItems:i,type:"tooltip"})));var t,e,i}getTitle(t,e){const{callbacks:i}=e,s=i.beforeTitle.apply(this,[t]),n=i.title.apply(this,[t]),o=i.afterTitle.apply(this,[t]);let a=[];return a=No(a,Wo(s)),a=No(a,Wo(n)),a=No(a,Wo(o)),a}getBeforeBody(t,e){return qo(e.callbacks.beforeBody.apply(this,[t]))}getBody(t,e){const{callbacks:i}=e,s=[];return d(t,(t=>{const e={before:[],lines:[],after:[]},n=Ko(i,t);No(e.before,Wo(n.beforeLabel.call(this,t))),No(e.lines,n.label.call(this,t)),No(e.after,Wo(n.afterLabel.call(this,t))),s.push(e)})),s}getAfterBody(t,e){return qo(e.callbacks.afterBody.apply(this,[t]))}getFooter(t,e){const{callbacks:i}=e,s=i.beforeFooter.apply(this,[t]),n=i.footer.apply(this,[t]),o=i.afterFooter.apply(this,[t]);let a=[];return a=No(a,Wo(s)),a=No(a,Wo(n)),a=No(a,Wo(o)),a}_createItems(t){const e=this._active,i=this.chart.data,s=[],n=[],o=[];let a,r,l=[];for(a=0,r=e.length;at.filter(e,s,n,i)))),t.itemSort&&(l=l.sort(((e,s)=>t.itemSort(e,s,i)))),d(l,(e=>{const i=Ko(t.callbacks,e);s.push(i.labelColor.call(this,e)),n.push(i.labelPointStyle.call(this,e)),o.push(i.labelTextColor.call(this,e))})),this.labelColors=s,this.labelPointStyles=n,this.labelTextColors=o,this.dataPoints=l,l}update(t,e){const i=this.options.setContext(this.getContext()),s=this._active;let n,o=[];if(s.length){const t=Bo[i.position].call(this,s,this._eventPosition);o=this._createItems(i),this.title=this.getTitle(o,i),this.beforeBody=this.getBeforeBody(o,i),this.body=this.getBody(o,i),this.afterBody=this.getAfterBody(o,i),this.footer=this.getFooter(o,i);const e=this._size=Ho(this,i),a=Object.assign({},t,e),r=Yo(this.chart,i,a),l=Uo(i,a,r,this.chart);this.xAlign=r.xAlign,this.yAlign=r.yAlign,n={opacity:1,x:l.x,y:l.y,width:e.width,height:e.height,caretX:t.x,caretY:t.y}}else 0!==this.opacity&&(n={opacity:0});this._tooltipItems=o,this.$context=void 0,n&&this._resolveAnimations().update(this,n),t&&i.external&&i.external.call(this,{chart:this.chart,tooltip:this,replay:e})}drawCaret(t,e,i,s){const n=this.getCaretPosition(t,i,s);e.lineTo(n.x1,n.y1),e.lineTo(n.x2,n.y2),e.lineTo(n.x3,n.y3)}getCaretPosition(t,e,i){const{xAlign:s,yAlign:n}=this,{caretSize:o,cornerRadius:a}=i,{topLeft:r,topRight:l,bottomLeft:h,bottomRight:c}=gi(a),{x:d,y:u}=t,{width:f,height:g}=e;let p,m,b,x,_,y;return"center"===n?(_=u+g/2,"left"===s?(p=d,m=p-o,x=_+o,y=_-o):(p=d+f,m=p+o,x=_-o,y=_+o),b=p):(m="left"===s?d+Math.max(r,h)+o:"right"===s?d+f-Math.max(l,c)-o:this.caretX,"top"===n?(x=u,_=x-o,p=m-o,b=m+o):(x=u+g,_=x+o,p=m+o,b=m-o),y=x),{x1:p,x2:m,x3:b,y1:x,y2:_,y3:y}}drawTitle(t,e,i){const s=this.title,n=s.length;let o,a,r;if(n){const l=yi(i.rtl,this.x,this.width);for(t.x=Xo(this,i.titleAlign,i),e.textAlign=l.textAlign(i.titleAlign),e.textBaseline="middle",o=mi(i.titleFont),a=i.titleSpacing,e.fillStyle=i.titleColor,e.font=o.string,r=0;r0!==t))?(t.beginPath(),t.fillStyle=o.multiKeyBackground,Le(t,{x:e,y:p,w:h,h:l,radius:r}),t.fill(),t.stroke(),t.fillStyle=a.backgroundColor,t.beginPath(),Le(t,{x:i,y:p+1,w:h-2,h:l-2,radius:r}),t.fill()):(t.fillStyle=o.multiKeyBackground,t.fillRect(e,p,h,l),t.strokeRect(e,p,h,l),t.fillStyle=a.backgroundColor,t.fillRect(i,p+1,h-2,l-2))}t.fillStyle=this.labelTextColors[i]}drawBody(t,e,i){const{body:s}=this,{bodySpacing:n,bodyAlign:o,displayColors:a,boxHeight:r,boxWidth:l,boxPadding:h}=i,c=mi(i.bodyFont);let u=c.lineHeight,f=0;const g=yi(i.rtl,this.x,this.width),p=function(i){e.fillText(i,g.x(t.x+f),t.y+u/2),t.y+=u+n},m=g.textAlign(o);let b,x,_,y,v,w,M;for(e.textAlign=o,e.textBaseline="middle",e.font=c.string,t.x=Xo(this,m,i),e.fillStyle=i.bodyColor,d(this.beforeBody,p),f=a&&"right"!==m?"center"===o?l/2+h:l+2+h:0,y=0,w=s.length;y0&&e.stroke()}_updateAnimationTarget(t){const e=this.chart,i=this.$animations,s=i&&i.x,n=i&&i.y;if(s||n){const i=Bo[t.position].call(this,this._active,this._eventPosition);if(!i)return;const o=this._size=Ho(this,t),a=Object.assign({},i,this._size),r=Yo(e,t,a),l=Uo(t,a,r,e);s._to===l.x&&n._to===l.y||(this.xAlign=r.xAlign,this.yAlign=r.yAlign,this.width=o.width,this.height=o.height,this.caretX=i.x,this.caretY=i.y,this._resolveAnimations().update(this,l))}}_willRender(){return!!this.opacity}draw(t){const e=this.options.setContext(this.getContext());let i=this.opacity;if(!i)return;this._updateAnimationTarget(e);const s={width:this.width,height:this.height},n={x:this.x,y:this.y};i=Math.abs(i)<.001?0:i;const o=pi(e.padding),a=this.title.length||this.beforeBody.length||this.body.length||this.afterBody.length||this.footer.length;e.enabled&&a&&(t.save(),t.globalAlpha=i,this.drawBackground(n,t,s,e),vi(t,e.textDirection),n.y+=o.top,this.drawTitle(n,t,e),this.drawBody(n,t,e),this.drawFooter(n,t,e),wi(t,e.textDirection),t.restore())}getActiveElements(){return this._active||[]}setActiveElements(t,e){const i=this._active,s=t.map((({datasetIndex:t,index:e})=>{const i=this.chart.getDatasetMeta(t);if(!i)throw new Error("Cannot find a dataset at index "+t);return{datasetIndex:t,element:i.data[e],index:e}})),n=!u(i,s),o=this._positionChanged(s,e);(n||o)&&(this._active=s,this._eventPosition=e,this._ignoreReplayEvents=!0,this.update(!0))}handleEvent(t,e,i=!0){if(e&&this._ignoreReplayEvents)return!1;this._ignoreReplayEvents=!1;const s=this.options,n=this._active||[],o=this._getActiveElements(t,n,e,i),a=this._positionChanged(o,t),r=e||!u(o,n)||a;return r&&(this._active=o,(s.enabled||s.external)&&(this._eventPosition={x:t.x,y:t.y},this.update(!0,e))),r}_getActiveElements(t,e,i,s){const n=this.options;if("mouseout"===t.type)return[];if(!s)return e;const o=this.chart.getElementsAtEventForMode(t,n.mode,n,i);return n.reverse&&o.reverse(),o}_positionChanged(t,e){const{caretX:i,caretY:s,options:n}=this,o=Bo[n.position].call(this,t,e);return!1!==o&&(i!==o.x||s!==o.y)}}Go.positioners=Bo;var Zo={id:"tooltip",_element:Go,positioners:Bo,afterInit(t,e,i){i&&(t.tooltip=new Go({chart:t,options:i}))},beforeUpdate(t,e,i){t.tooltip&&t.tooltip.initialize(i)},reset(t,e,i){t.tooltip&&t.tooltip.initialize(i)},afterDraw(t){const e=t.tooltip;if(e&&e._willRender()){const i={tooltip:e};if(!1===t.notifyPlugins("beforeTooltipDraw",i))return;e.draw(t.ctx),t.notifyPlugins("afterTooltipDraw",i)}},afterEvent(t,e){if(t.tooltip){const i=e.replay;t.tooltip.handleEvent(e.event,i,e.inChartArea)&&(e.changed=!0)}},defaults:{enabled:!0,external:null,position:"average",backgroundColor:"rgba(0,0,0,0.8)",titleColor:"#fff",titleFont:{weight:"bold"},titleSpacing:2,titleMarginBottom:6,titleAlign:"left",bodyColor:"#fff",bodySpacing:2,bodyFont:{},bodyAlign:"left",footerColor:"#fff",footerSpacing:2,footerMarginTop:6,footerFont:{weight:"bold"},footerAlign:"left",padding:6,caretPadding:2,caretSize:5,cornerRadius:6,boxHeight:(t,e)=>e.bodyFont.size,boxWidth:(t,e)=>e.bodyFont.size,multiKeyBackground:"#fff",displayColors:!0,boxPadding:0,borderColor:"rgba(0,0,0,0)",borderWidth:0,animation:{duration:400,easing:"easeOutQuart"},animations:{numbers:{type:"number",properties:["x","y","width","height","caretX","caretY"]},opacity:{easing:"linear",duration:200}},callbacks:{beforeTitle:t,title(t){if(t.length>0){const e=t[0],i=e.chart.data.labels,s=i?i.length:0;if(this&&this.options&&"dataset"===this.options.mode)return e.dataset.label||"";if(e.label)return e.label;if(s>0&&e.dataIndex"filter"!==t&&"itemSort"!==t&&"external"!==t,_indexable:!1,callbacks:{_scriptable:!1,_indexable:!1},animation:{_fallback:!1},animations:{_fallback:"animation"}},additionalOptionScopes:["interaction"]},Jo=Object.freeze({__proto__:null,Decimation:go,Filler:To,Legend:Ro,SubTitle:Vo,Title:zo,Tooltip:Zo});function Qo(t,e,i,s){const n=t.indexOf(e);if(-1===n)return((t,e,i,s)=>("string"==typeof e?(i=t.push(e)-1,s.unshift({index:i,label:e})):isNaN(e)&&(i=null),i))(t,e,i,s);return n!==t.lastIndexOf(e)?i:n}class ta extends $s{constructor(t){super(t),this._startValue=void 0,this._valueRange=0,this._addedLabels=[]}init(t){const e=this._addedLabels;if(e.length){const t=this.getLabels();for(const{index:i,label:s}of e)t[i]===s&&t.splice(i,1);this._addedLabels=[]}super.init(t)}parse(t,e){if(i(t))return null;const s=this.getLabels();return((t,e)=>null===t?null:Z(Math.round(t),0,e))(e=isFinite(e)&&s[e]===t?e:Qo(s,t,r(e,t),this._addedLabels),s.length-1)}determineDataLimits(){const{minDefined:t,maxDefined:e}=this.getUserBounds();let{min:i,max:s}=this.getMinMax(!0);"ticks"===this.options.bounds&&(t||(i=0),e||(s=this.getLabels().length-1)),this.min=i,this.max=s}buildTicks(){const t=this.min,e=this.max,i=this.options.offset,s=[];let n=this.getLabels();n=0===t&&e===n.length-1?n:n.slice(t,e+1),this._valueRange=Math.max(n.length-(i?0:1),1),this._startValue=this.min-(i?.5:0);for(let i=t;i<=e;i++)s.push({value:i});return s}getLabelForValue(t){const e=this.getLabels();return t>=0&&te.length-1?null:this.getPixelForValue(e[t].value)}getValueForPixel(t){return Math.round(this._startValue+this.getDecimalForPixel(t)*this._valueRange)}getBasePixel(){return this.bottom}}function ea(t,e,{horizontal:i,minRotation:s}){const n=H(s),o=(i?Math.sin(n):Math.cos(n))||.001,a=.75*e*(""+t).length;return Math.min(e/o,a)}ta.id="category",ta.defaults={ticks:{callback:ta.prototype.getLabelForValue}};class ia extends $s{constructor(t){super(t),this.start=void 0,this.end=void 0,this._startValue=void 0,this._endValue=void 0,this._valueRange=0}parse(t,e){return i(t)||("number"==typeof t||t instanceof Number)&&!isFinite(+t)?null:+t}handleTickRangeOptions(){const{beginAtZero:t}=this.options,{minDefined:e,maxDefined:i}=this.getUserBounds();let{min:s,max:n}=this;const o=t=>s=e?s:t,a=t=>n=i?n:t;if(t){const t=z(s),e=z(n);t<0&&e<0?a(0):t>0&&e>0&&o(0)}if(s===n){let e=1;(n>=Number.MAX_SAFE_INTEGER||s<=Number.MIN_SAFE_INTEGER)&&(e=Math.abs(.05*n)),a(n+e),t||o(s-e)}this.min=s,this.max=n}getTickLimit(){const t=this.options.ticks;let e,{maxTicksLimit:i,stepSize:s}=t;return s?(e=Math.ceil(this.max/s)-Math.floor(this.min/s)+1,e>1e3&&(console.warn(`scales.${this.id}.ticks.stepSize: ${s} would result generating up to ${e} ticks. Limiting to 1000.`),e=1e3)):(e=this.computeTickLimit(),i=i||11),i&&(e=Math.min(i,e)),e}computeTickLimit(){return Number.POSITIVE_INFINITY}buildTicks(){const t=this.options,e=t.ticks;let s=this.getTickLimit();s=Math.max(2,s);const n=function(t,e){const s=[],{bounds:n,step:o,min:a,max:r,precision:l,count:h,maxTicks:c,maxDigits:d,includeBounds:u}=t,f=o||1,g=c-1,{min:p,max:m}=e,b=!i(a),x=!i(r),_=!i(h),y=(m-p)/(d+1);let v,w,M,k,S=F((m-p)/g/f)*f;if(S<1e-14&&!b&&!x)return[{value:p},{value:m}];k=Math.ceil(m/S)-Math.floor(p/S),k>g&&(S=F(k*S/g/f)*f),i(l)||(v=Math.pow(10,l),S=Math.ceil(S*v)/v),"ticks"===n?(w=Math.floor(p/S)*S,M=Math.ceil(m/S)*S):(w=p,M=m),b&&x&&o&&W((r-a)/o,S/1e3)?(k=Math.round(Math.min((r-a)/S,c)),S=(r-a)/k,w=a,M=r):_?(w=b?a:w,M=x?r:M,k=h-1,S=(M-w)/k):(k=(M-w)/S,k=N(k,Math.round(k),S/1e3)?Math.round(k):Math.ceil(k));const P=Math.max(Y(S),Y(w));v=Math.pow(10,i(l)?P:l),w=Math.round(w*v)/v,M=Math.round(M*v)/v;let D=0;for(b&&(u&&w!==a?(s.push({value:a}),w0?i:null;this._zero=!0}determineDataLimits(){const{min:t,max:e}=this.getMinMax(!0);this.min=o(t)?Math.max(0,t):null,this.max=o(e)?Math.max(0,e):null,this.options.beginAtZero&&(this._zero=!0),this.handleTickRangeOptions()}handleTickRangeOptions(){const{minDefined:t,maxDefined:e}=this.getUserBounds();let i=this.min,s=this.max;const n=e=>i=t?i:e,o=t=>s=e?s:t,a=(t,e)=>Math.pow(10,Math.floor(I(t))+e);i===s&&(i<=0?(n(1),o(10)):(n(a(i,-1)),o(a(s,1)))),i<=0&&n(a(s,-1)),s<=0&&o(a(i,1)),this._zero&&this.min!==this._suggestedMin&&i===a(this.min,0)&&n(a(i,-1)),this.min=i,this.max=s}buildTicks(){const t=this.options,e=function(t,e){const i=Math.floor(I(e.max)),s=Math.ceil(e.max/Math.pow(10,i)),n=[];let o=a(t.min,Math.pow(10,Math.floor(I(e.min)))),r=Math.floor(I(o)),l=Math.floor(o/Math.pow(10,r)),h=r<0?Math.pow(10,Math.abs(r)):1;do{n.push({value:o,major:na(o)}),++l,10===l&&(l=1,++r,h=r>=0?1:h),o=Math.round(l*Math.pow(10,r)*h)/h}while(rn?{start:e-i,end:e}:{start:e,end:e+i}}function la(t){const e={l:t.left+t._padding.left,r:t.right-t._padding.right,t:t.top+t._padding.top,b:t.bottom-t._padding.bottom},i=Object.assign({},e),n=[],o=[],a=t._pointLabels.length,r=t.options.pointLabels,l=r.centerPointLabels?D/a:0;for(let u=0;ue.r&&(r=(s.end-e.r)/o,t.r=Math.max(t.r,e.r+r)),n.starte.b&&(l=(n.end-e.b)/a,t.b=Math.max(t.b,e.b+l))}function ca(t){return 0===t||180===t?"center":t<180?"left":"right"}function da(t,e,i){return"right"===i?t-=e:"center"===i&&(t-=e/2),t}function ua(t,e,i){return 90===i||270===i?t-=e/2:(i>270||i<90)&&(t-=e),t}function fa(t,e,i,s){const{ctx:n}=t;if(i)n.arc(t.xCenter,t.yCenter,e,0,O);else{let i=t.getPointPosition(0,e);n.moveTo(i.x,i.y);for(let o=1;o{const i=c(this.options.pointLabels.callback,[t,e],this);return i||0===i?i:""})).filter(((t,e)=>this.chart.getDataVisibility(e)))}fit(){const t=this.options;t.display&&t.pointLabels.display?la(this):this.setCenterPoint(0,0,0,0)}setCenterPoint(t,e,i,s){this.xCenter+=Math.floor((t-e)/2),this.yCenter+=Math.floor((i-s)/2),this.drawingArea-=Math.min(this.drawingArea/2,Math.max(t,e,i,s))}getIndexAngle(t){return K(t*(O/(this._pointLabels.length||1))+H(this.options.startAngle||0))}getDistanceFromCenterForValue(t){if(i(t))return NaN;const e=this.drawingArea/(this.max-this.min);return this.options.reverse?(this.max-t)*e:(t-this.min)*e}getValueForDistanceFromCenter(t){if(i(t))return NaN;const e=t/(this.drawingArea/(this.max-this.min));return this.options.reverse?this.max-e:this.min+e}getPointLabelContext(t){const e=this._pointLabels||[];if(t>=0&&t=0;o--){const e=n.setContext(t.getPointLabelContext(o)),a=mi(e.font),{x:r,y:l,textAlign:h,left:c,top:d,right:u,bottom:f}=t._pointLabelItems[o],{backdropColor:g}=e;if(!i(g)){const t=gi(e.borderRadius),i=pi(e.backdropPadding);s.fillStyle=g;const n=c-i.left,o=d-i.top,a=u-c+i.width,r=f-d+i.height;Object.values(t).some((t=>0!==t))?(s.beginPath(),Le(s,{x:n,y:o,w:a,h:r,radius:t}),s.fill()):s.fillRect(n,o,a,r)}Ae(s,t._pointLabels[o],r,l+a.lineHeight/2,a,{color:e.color,textAlign:h,textBaseline:"middle"})}}(this,o),n.display&&this.ticks.forEach(((t,e)=>{if(0!==e){r=this.getDistanceFromCenterForValue(t.value);!function(t,e,i,s){const n=t.ctx,o=e.circular,{color:a,lineWidth:r}=e;!o&&!s||!a||!r||i<0||(n.save(),n.strokeStyle=a,n.lineWidth=r,n.setLineDash(e.borderDash),n.lineDashOffset=e.borderDashOffset,n.beginPath(),fa(t,i,o,s),n.closePath(),n.stroke(),n.restore())}(this,n.setContext(this.getContext(e-1)),r,o)}})),s.display){for(t.save(),a=o-1;a>=0;a--){const i=s.setContext(this.getPointLabelContext(a)),{color:n,lineWidth:o}=i;o&&n&&(t.lineWidth=o,t.strokeStyle=n,t.setLineDash(i.borderDash),t.lineDashOffset=i.borderDashOffset,r=this.getDistanceFromCenterForValue(e.ticks.reverse?this.min:this.max),l=this.getPointPosition(a,r),t.beginPath(),t.moveTo(this.xCenter,this.yCenter),t.lineTo(l.x,l.y),t.stroke())}t.restore()}}drawBorder(){}drawLabels(){const t=this.ctx,e=this.options,i=e.ticks;if(!i.display)return;const s=this.getIndexAngle(0);let n,o;t.save(),t.translate(this.xCenter,this.yCenter),t.rotate(s),t.textAlign="center",t.textBaseline="middle",this.ticks.forEach(((s,a)=>{if(0===a&&!e.reverse)return;const r=i.setContext(this.getContext(a)),l=mi(r.font);if(n=this.getDistanceFromCenterForValue(this.ticks[a].value),r.showLabelBackdrop){t.font=l.string,o=t.measureText(s.label).width,t.fillStyle=r.backdropColor;const e=pi(r.backdropPadding);t.fillRect(-o/2-e.left,-n-l.size/2-e.top,o+e.width,l.size+e.height)}Ae(t,s.label,0,-n,l,{color:r.color})})),t.restore()}drawTitle(){}}ga.id="radialLinear",ga.defaults={display:!0,animate:!0,position:"chartArea",angleLines:{display:!0,lineWidth:1,borderDash:[],borderDashOffset:0},grid:{circular:!1},startAngle:0,ticks:{showLabelBackdrop:!0,callback:Is.formatters.numeric},pointLabels:{backdropColor:void 0,backdropPadding:2,display:!0,font:{size:10},callback:t=>t,padding:5,centerPointLabels:!1}},ga.defaultRoutes={"angleLines.color":"borderColor","pointLabels.color":"color","ticks.color":"color"},ga.descriptors={angleLines:{_fallback:"grid"}};const pa={millisecond:{common:!0,size:1,steps:1e3},second:{common:!0,size:1e3,steps:60},minute:{common:!0,size:6e4,steps:60},hour:{common:!0,size:36e5,steps:24},day:{common:!0,size:864e5,steps:30},week:{common:!1,size:6048e5,steps:4},month:{common:!0,size:2628e6,steps:12},quarter:{common:!1,size:7884e6,steps:4},year:{common:!0,size:3154e7}},ma=Object.keys(pa);function ba(t,e){return t-e}function xa(t,e){if(i(e))return null;const s=t._adapter,{parser:n,round:a,isoWeekday:r}=t._parseOpts;let l=e;return"function"==typeof n&&(l=n(l)),o(l)||(l="string"==typeof n?s.parse(l,n):s.parse(l)),null===l?null:(a&&(l="week"!==a||!B(r)&&!0!==r?s.startOf(l,a):s.startOf(l,"isoWeek",r)),+l)}function _a(t,e,i,s){const n=ma.length;for(let o=ma.indexOf(t);o=e?i[s]:i[n]]=!0}}else t[e]=!0}function va(t,e,i){const s=[],n={},o=e.length;let a,r;for(a=0;a=0&&(e[l].major=!0);return e}(t,s,n,i):s}class wa extends $s{constructor(t){super(t),this._cache={data:[],labels:[],all:[]},this._unit="day",this._majorUnit=void 0,this._offsets={},this._normalized=!1,this._parseOpts=void 0}init(t,e){const i=t.time||(t.time={}),s=this._adapter=new wn._date(t.adapters.date);s.init(e),b(i.displayFormats,s.formats()),this._parseOpts={parser:i.parser,round:i.round,isoWeekday:i.isoWeekday},super.init(t),this._normalized=e.normalized}parse(t,e){return void 0===t?null:xa(this,t)}beforeLayout(){super.beforeLayout(),this._cache={data:[],labels:[],all:[]}}determineDataLimits(){const t=this.options,e=this._adapter,i=t.time.unit||"day";let{min:s,max:n,minDefined:a,maxDefined:r}=this.getUserBounds();function l(t){a||isNaN(t.min)||(s=Math.min(s,t.min)),r||isNaN(t.max)||(n=Math.max(n,t.max))}a&&r||(l(this._getLabelBounds()),"ticks"===t.bounds&&"labels"===t.ticks.source||l(this.getMinMax(!1))),s=o(s)&&!isNaN(s)?s:+e.startOf(Date.now(),i),n=o(n)&&!isNaN(n)?n:+e.endOf(Date.now(),i)+1,this.min=Math.min(s,n-1),this.max=Math.max(s+1,n)}_getLabelBounds(){const t=this.getLabelTimestamps();let e=Number.POSITIVE_INFINITY,i=Number.NEGATIVE_INFINITY;return t.length&&(e=t[0],i=t[t.length-1]),{min:e,max:i}}buildTicks(){const t=this.options,e=t.time,i=t.ticks,s="labels"===i.source?this.getLabelTimestamps():this._generate();"ticks"===t.bounds&&s.length&&(this.min=this._userMin||s[0],this.max=this._userMax||s[s.length-1]);const n=this.min,o=st(s,n,this.max);return this._unit=e.unit||(i.autoSkip?_a(e.minUnit,this.min,this.max,this._getLabelCapacity(n)):function(t,e,i,s,n){for(let o=ma.length-1;o>=ma.indexOf(i);o--){const i=ma[o];if(pa[i].common&&t._adapter.diff(n,s,i)>=e-1)return i}return ma[i?ma.indexOf(i):0]}(this,o.length,e.minUnit,this.min,this.max)),this._majorUnit=i.major.enabled&&"year"!==this._unit?function(t){for(let e=ma.indexOf(t)+1,i=ma.length;e+t.value)))}initOffsets(t){let e,i,s=0,n=0;this.options.offset&&t.length&&(e=this.getDecimalForValue(t[0]),s=1===t.length?1-e:(this.getDecimalForValue(t[1])-e)/2,i=this.getDecimalForValue(t[t.length-1]),n=1===t.length?i:(i-this.getDecimalForValue(t[t.length-2]))/2);const o=t.length<3?.5:.25;s=Z(s,0,o),n=Z(n,0,o),this._offsets={start:s,end:n,factor:1/(s+1+n)}}_generate(){const t=this._adapter,e=this.min,i=this.max,s=this.options,n=s.time,o=n.unit||_a(n.minUnit,e,i,this._getLabelCapacity(e)),a=r(n.stepSize,1),l="week"===o&&n.isoWeekday,h=B(l)||!0===l,c={};let d,u,f=e;if(h&&(f=+t.startOf(f,"isoWeek",l)),f=+t.startOf(f,h?"day":o),t.diff(i,e,o)>1e5*a)throw new Error(e+" and "+i+" are too far apart with stepSize of "+a+" "+o);const g="data"===s.ticks.source&&this.getDataTimestamps();for(d=f,u=0;dt-e)).map((t=>+t))}getLabelForValue(t){const e=this._adapter,i=this.options.time;return i.tooltipFormat?e.format(t,i.tooltipFormat):e.format(t,i.displayFormats.datetime)}_tickFormatFunction(t,e,i,s){const n=this.options,o=n.time.displayFormats,a=this._unit,r=this._majorUnit,l=a&&o[a],h=r&&o[r],d=i[e],u=r&&h&&d&&d.major,f=this._adapter.format(t,s||(u?h:l)),g=n.ticks.callback;return g?c(g,[f,e,i],this):f}generateTickLabels(t){let e,i,s;for(e=0,i=t.length;e0?a:1}getDataTimestamps(){let t,e,i=this._cache.data||[];if(i.length)return i;const s=this.getMatchingVisibleMetas();if(this._normalized&&s.length)return this._cache.data=s[0].controller.getAllParsedValues(this);for(t=0,e=s.length;t=t[r].pos&&e<=t[l].pos&&({lo:r,hi:l}=et(t,"pos",e)),({pos:s,time:o}=t[r]),({pos:n,time:a}=t[l])):(e>=t[r].time&&e<=t[l].time&&({lo:r,hi:l}=et(t,"time",e)),({time:s,pos:o}=t[r]),({time:n,pos:a}=t[l]));const h=n-s;return h?o+(a-o)*(e-s)/h:o}wa.id="time",wa.defaults={bounds:"data",adapters:{},time:{parser:!1,unit:!1,round:!1,isoWeekday:!1,minUnit:"millisecond",displayFormats:{}},ticks:{source:"auto",major:{enabled:!1}}};class ka extends wa{constructor(t){super(t),this._table=[],this._minPos=void 0,this._tableRange=void 0}initOffsets(){const t=this._getTimestampsForTable(),e=this._table=this.buildLookupTable(t);this._minPos=Ma(e,this.min),this._tableRange=Ma(e,this.max)-this._minPos,super.initOffsets(t)}buildLookupTable(t){const{min:e,max:i}=this,s=[],n=[];let o,a,r,l,h;for(o=0,a=t.length;o=e&&l<=i&&s.push(l);if(s.length<2)return[{time:e,pos:0},{time:i,pos:1}];for(o=0,a=s.length;o { + const key = element.getAttribute('data-i18n'); + if (key) { + element.textContent = t(key); + } + }); + + // Translate placeholders + const placeholderElements = document.querySelectorAll('[data-i18n-placeholder]'); + placeholderElements.forEach(element => { + const key = element.getAttribute('data-i18n-placeholder'); + if (key) { + element.placeholder = t(key); + } + }); + + // Translate titles/tooltips + const titleElements = document.querySelectorAll('[data-i18n-title]'); + titleElements.forEach(element => { + const key = element.getAttribute('data-i18n-title'); + if (key) { + element.title = t(key); + } + }); + + // Translate aria-labels + const ariaElements = document.querySelectorAll('[data-i18n-aria]'); + ariaElements.forEach(element => { + const key = element.getAttribute('data-i18n-aria'); + if (key) { + element.setAttribute('aria-label', t(key)); + } + }); + + // Translate tooltips + const tooltipElements = document.querySelectorAll('[data-i18n-tooltip]'); + tooltipElements.forEach(element => { + const key = element.getAttribute('data-i18n-tooltip'); + if (key) { + element.setAttribute('data-tooltip', t(key)); + } + }); +} + +/** + * Show notification message + */ +function showNotification(message, type = 'info') { + // Create notification element if it doesn't exist + let notification = document.getElementById('i18nNotification'); + if (!notification) { + notification = document.createElement('div'); + notification.id = 'i18nNotification'; + notification.style.cssText = ` + position: fixed; + top: 20px; + right: 20px; + padding: 12px 20px; + border-radius: 8px; + color: white; + font-weight: bold; + z-index: 10000; + transition: all 0.3s ease; + transform: translateX(100%); + `; + document.body.appendChild(notification); + } + + // Set color based on type + const colors = { + success: '#28a745', + error: '#dc3545', + warning: '#ffc107', + info: '#007bff' + }; + + notification.style.backgroundColor = colors[type] || colors.info; + notification.textContent = message; + notification.style.transform = 'translateX(0)'; + + // Hide after 3 seconds + setTimeout(() => { + notification.style.transform = 'translateX(100%)'; + }, 3000); +} + +/** + * Create language selector dropdown + */ +function createLanguageSelector(containerId) { + const container = document.getElementById(containerId); + if (!container) return; + + const selector = document.createElement('select'); + selector.id = 'languageSelector'; + selector.style.cssText = ` + width: 100%; + padding: 14px 18px; + border: 2px solid #dee2e6; + border-radius: 10px; + background: white; + color: #495057; + font-size: 0.95rem; + font-weight: 500; + cursor: pointer; + transition: all 0.3s ease; + box-shadow: 0 3px 6px rgba(0, 0, 0, 0.1); + margin-bottom: 12px; + `; + + // Add options + const languages = [ + { code: 'sl', name: '🇸🇮 Slovenščina', flag: '🇸🇮' }, + { code: 'en', name: '🇬🇧 English', flag: '🇬🇧' } + ]; + + languages.forEach(lang => { + const option = document.createElement('option'); + option.value = lang.code; + option.textContent = lang.name; + selector.appendChild(option); + }); + + selector.value = currentLanguage; + + // Add event listeners + selector.addEventListener('change', (e) => { + changeLanguage(e.target.value); + }); + + // Add hover effects to match nav-link styling + selector.addEventListener('mouseenter', () => { + selector.style.background = '#f8f9fa'; + selector.style.borderColor = '#007bff'; + selector.style.color = '#007bff'; + selector.style.transform = 'translateY(-2px)'; + selector.style.boxShadow = '0 5px 12px rgba(0, 123, 255, 0.15)'; + }); + + selector.addEventListener('mouseleave', () => { + selector.style.background = 'white'; + selector.style.borderColor = '#dee2e6'; + selector.style.color = '#495057'; + selector.style.transform = 'translateY(0)'; + selector.style.boxShadow = '0 3px 6px rgba(0, 0, 0, 0.1)'; + }); + + container.appendChild(selector); +} + +// Initialize when DOM is loaded +document.addEventListener('DOMContentLoaded', initI18n); + +// Export functions for global use +window.t = t; +window.changeLanguage = changeLanguage; +window.createLanguageSelector = createLanguageSelector; \ No newline at end of file diff --git a/streams.json b/streams.json new file mode 100644 index 0000000..392315c --- /dev/null +++ b/streams.json @@ -0,0 +1,39 @@ +{ + "1": { + "number": 1, + "title": "Stream 1", + "url": "http://192.168.0.103:9081", + "active": true + }, + "2": { + "number": 2, + "title": "Stream 2", + "url": "http://192.168.0.103:9082", + "active": true + }, + "3": { + "number": 3, + "title": "Stream 3", + "url": "http://192.168.0.103:9083", + "active": true + }, + "4": { + "number": 4, + "title": "Stream 4", + "url": "http://192.168.0.103:9084", + "active": true + }, + "5": { + "number": 5, + "title": "Stream 5", + "url": "http://192.168.0.103:9085", + "active": true + }, + "6": { + "number": 6, + "title": "Stream 6", + "url": "http://192.168.0.103:9086", + "active": true + }, + "_last_updated": 1757351904.1407912 +} \ No newline at end of file diff --git a/templates/draft.html b/templates/draft.html index 8d30a46..3419101 100644 --- a/templates/draft.html +++ b/templates/draft.html @@ -2,7 +2,7 @@ - Tournament Draft + Tournament Draft @@ -439,14 +585,14 @@ - +
+ +
-

Camera Settings

+

Nastavitve

{% if settings.tournament_active %}
-

🏆 Active Tournament

- +

🏆 Aktiven Turnir

+
-

Current Round: {{ settings.current_round }} of {{ settings.total_rounds }}

-

Camera cards show current round players

+

Trenutni Krog: {{ settings.current_round }} od {{ settings.total_rounds }}

+

Kartice kamer prikazujejo igralce trenutnega kroga

- +
{% endif %} +
- {% if not settings.tournament_active %} -

Navigation

- 🏆 Tournament Mode -

Data & Analysis

- 👤 Player Analysis - 📚 View Archive - {% endif %} +

Turnirji

+ 🏆 Način Turnirja + 👤 Analiza Igralcev + 📚 Arhiv +
+ + +
+

Sistem

+
+ +
+
{% if not settings.tournament_active %} +
-

Display Options

- +

Kamera

+
-
-
-
-

Camera Titles

- {% for i in range(1, 7) %} -
- - + +
+
Naslovi Kamer
+ {% for i in range(1, 7) %} +
+ + +
+ {% endfor %}
- {% endfor %}
{% endif %} + + +
+ Version 1.0.0 +
{% for i in range(1, 7) %} -
-
{{ settings.camera_titles[i|string] }}
+
+
+
{{ i }}
+
{{ settings.camera_titles[i|string] }}
+
Camera Stream {{ i }}
@@ -561,315 +728,605 @@ const tournamentActive = {{ 'true' if settings.tournament_active else 'false' }}; let currentRound = {{ settings.current_round if settings.tournament_active else 1 }}; const totalRounds = {{ settings.total_rounds if settings.tournament_active else 1 }}; + + // Remote control state + const remoteControlled = {{ 'true' if settings.remote_controlled else 'false' }}; + let remotePollingInterval = null; + let lastRemoteUpdate = '{{ settings.dashboard_state.last_updated if settings.dashboard_state else "" }}'; + + // DOM Elements - declare early to avoid issues + let menuButton, settingsPanel, overlay, closeBtn; + + // Initialize DOM elements + function initializeDOMElements() { + menuButton = document.getElementById('menuButton'); + settingsPanel = document.getElementById('settingsPanel'); + overlay = document.getElementById('overlay'); + closeBtn = document.getElementById('closeBtn'); + + console.log('DOM Elements initialized:', { + menuButton: !!menuButton, + settingsPanel: !!settingsPanel, + overlay: !!overlay, + closeBtn: !!closeBtn + }); + } + + // Settings panel functionality + function openSettings() { + console.log('Opening settings panel'); + if (settingsPanel && overlay) { + settingsPanel.classList.add('active'); + overlay.classList.add('active'); + } else { + console.error('Settings panel elements not found'); + } + } + + function closeSettings() { + console.log('Closing settings panel'); + if (settingsPanel && overlay) { + settingsPanel.classList.remove('active'); + overlay.classList.remove('active'); + } + } + + // Remote control indicator + function createRemoteIndicator() { + if (remoteControlled) { + const indicator = document.createElement('div'); + indicator.id = 'remoteIndicator'; + indicator.innerHTML = ` +
+
+ 📱 Remote Controlled +
+ + `; + document.body.appendChild(indicator); + } + } + + // Start remote control polling + function startRemotePolling() { + if (!remoteControlled) return; + + if (remotePollingInterval) { + clearInterval(remotePollingInterval); + } + + remotePollingInterval = setInterval(async () => { + try { + const response = await fetch('/api/remote/check_updates'); + if (response.ok) { + const data = await response.json(); + + if (data.needs_refresh) { + console.log('🔄 Remote refresh requested'); + showRemoteUpdateNotification('Dashboard refreshed remotely'); + setTimeout(() => { + window.location.reload(); + }, 1000); + } + + // Check if view should change + const currentView = data.current_view; + const currentPath = window.location.pathname; + + if (shouldRedirectToView(currentView, currentPath)) { + console.log(`🔄 Remote view change: ${currentView}`); + showRemoteUpdateNotification(`Switching to ${currentView} view`); + setTimeout(() => { + redirectToView(currentView, data.fullscreen_camera); + }, 1000); + } + + // Update last update time + if (data.last_updated && data.last_updated !== lastRemoteUpdate) { + lastRemoteUpdate = data.last_updated; + } + } + } catch (error) { + console.error('Remote polling error:', error); + // Don't show error to user, just continue polling + } + }, 2000); // Poll every 2 seconds + } + + // Check if current page should redirect based on remote command + function shouldRedirectToView(remoteView, currentPath) { + const viewPaths = { + 'cameras': '/', + 'draft': '/tournament/draft', + 'results': '/results', + 'calculator': '/results/calculator', + 'fullscreen': '/fullscreen/' + }; + + // If we're on the home page and remote wants cameras, no change needed + if (remoteView === 'cameras' && currentPath === '/') { + return false; + } + + // If we're on fullscreen and remote wants fullscreen, no change needed + if (remoteView === 'fullscreen' && currentPath.startsWith('/fullscreen/')) { + return false; + } + + // Check if current path matches the desired view + const targetPath = viewPaths[remoteView]; + if (targetPath && currentPath !== targetPath) { + return true; + } + + return false; + } + + // Redirect to the appropriate view + function redirectToView(view, fullscreenCamera = null) { + switch (view) { + case 'cameras': + window.location.href = '/'; + break; + case 'draft': + window.location.href = '/tournament/draft'; + break; + case 'results': + window.location.href = '/results'; + break; + case 'calculator': + window.location.href = '/results/calculator'; + break; + case 'fullscreen': + if (fullscreenCamera) { + window.location.href = `/fullscreen/${fullscreenCamera}`; + } + break; + default: + console.log('Unknown view:', view); + } + } + + // Show remote update notification - simplified for performance + function showRemoteUpdateNotification(message) { + const notification = document.createElement('div'); + notification.style.cssText = ` + position: fixed; + top: 20px; + right: 20px; + background: rgba(40, 167, 69, 0.9); + color: white; + padding: 12px 20px; + border-radius: 8px; + font-weight: bold; + z-index: 1001; + border: 1px solid rgba(255, 255, 255, 0.2); + box-shadow: 0 2px 10px rgba(0, 0, 0, 0.2); + `; + notification.textContent = `📱 ${message}`; + + document.body.appendChild(notification); + + // Remove after 3 seconds without heavy animations + setTimeout(() => { + if (notification.parentNode) { + notification.parentNode.removeChild(notification); + } + }, 3000); + } function updateRoundNavigation() { - if (tournamentActive) { - const prevBtn = document.getElementById('prevRoundBtn'); - const nextBtn = document.getElementById('nextRoundBtn'); - const currentRoundNum = document.getElementById('currentRoundNum'); - - if (prevBtn) prevBtn.disabled = currentRound <= 1; - if (nextBtn) nextBtn.disabled = currentRound >= totalRounds; - if (currentRoundNum) currentRoundNum.textContent = currentRound; - } + if (tournamentActive) { + const prevBtn = document.getElementById('prevRoundBtn'); + const nextBtn = document.getElementById('nextRoundBtn'); + const currentRoundNum = document.getElementById('currentRoundNum'); + + if (prevBtn) prevBtn.disabled = currentRound <= 1; + if (nextBtn) nextBtn.disabled = currentRound >= totalRounds; + if (currentRoundNum) currentRoundNum.textContent = currentRound; + } } // Change round function async function changeRound(direction) { - if (!tournamentActive) return; - - const newRound = currentRound + direction; - if (newRound < 1 || newRound > totalRounds) return; - - // Disable buttons during request - const prevBtn = document.getElementById('prevRoundBtn'); - const nextBtn = document.getElementById('nextRoundBtn'); - - if (prevBtn) prevBtn.disabled = true; - if (nextBtn) nextBtn.disabled = true; - - try { - const response = await fetch(`/api/tournament/round/${newRound}`, { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - } - }); + if (!tournamentActive) return; - if (response.ok) { - currentRound = newRound; - updateRoundNavigation(); - - // Reload page to get new round players - setTimeout(() => { - window.location.reload(); - }, 300); - } else { - console.error('Failed to change round'); - alert('Failed to change round. Please try again.'); - updateRoundNavigation(); + const newRound = currentRound + direction; + if (newRound < 1 || newRound > totalRounds) return; + + // Disable buttons during request + const prevBtn = document.getElementById('prevRoundBtn'); + const nextBtn = document.getElementById('nextRoundBtn'); + + if (prevBtn) prevBtn.disabled = true; + if (nextBtn) nextBtn.disabled = true; + + try { + const response = await fetch(`/api/tournament/round/${newRound}`, { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + } + }); + + if (response.ok) { + currentRound = newRound; + updateRoundNavigation(); + + // Show notification for remote users + if (remoteControlled) { + showRemoteUpdateNotification(`Round changed to ${newRound}`); + } + + // Reload page to get new round players + setTimeout(() => { + window.location.reload(); + }, 300); + } else { + console.error('Failed to change round'); + alert('Failed to change round. Please try again.'); + updateRoundNavigation(); + } + } catch (error) { + console.error('Error changing round:', error); + alert('Error changing round. Please try again.'); + updateRoundNavigation(); } - } catch (error) { - console.error('Error changing round:', error); - alert('Error changing round. Please try again.'); - updateRoundNavigation(); - } } // Debounce function function debounce(func, wait) { - let timeout; - return function executedFunction(...args) { - const later = () => { - clearTimeout(timeout); - func(...args); + let timeout; + return function executedFunction(...args) { + const later = () => { + clearTimeout(timeout); + func(...args); + }; + clearTimeout(timeout); + timeout = setTimeout(later, wait); }; - clearTimeout(timeout); - timeout = setTimeout(later, wait); - }; } // Save settings to server async function saveSettings(settingsUpdate) { - try { - const response = await fetch('/api/settings', { - method: 'POST', - headers: { - 'Content-Type': 'application/json', - }, - body: JSON.stringify(settingsUpdate) - }); - - if (response.ok) { - const result = await response.json(); - if (!tournamentActive) { - currentSettings = result.settings; - } + try { + const response = await fetch('/api/settings', { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify(settingsUpdate) + }); + + if (response.ok) { + const result = await response.json(); + if (!tournamentActive) { + currentSettings = result.settings; + } + } + } catch (error) { + console.error('Error saving settings:', error); } - } catch (error) { - console.error('Error saving settings:', error); - } } const debouncedSaveSettings = debounce(saveSettings, 500); - // Desktop-only functionality - mobile users are redirected - // Mobile device detection function isMobileDevice() { - return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) || - (window.innerWidth <= 1024 && window.innerHeight <= 1366) || - ('ontouchstart' in window) || - (navigator.maxTouchPoints > 0); + return /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) || + (window.innerWidth <= 1024 && window.innerHeight <= 1366) || + ('ontouchstart' in window) || + (navigator.maxTouchPoints > 0); } // Update card title function updateCardTitle(index) { - if (tournamentActive) return; - - const input = document.getElementById(`titleInput${index}`); - const cardTitle = document.getElementById(`cardTitle${index}`); - if (input && cardTitle) { - const newTitle = input.value || `Camera ${index}`; - cardTitle.textContent = newTitle; + if (tournamentActive) return; - if (event && event.target === input) { - currentSettings.camera_titles[index.toString()] = newTitle; - debouncedSaveSettings({ - camera_titles: { - [index.toString()]: newTitle + const input = document.getElementById(`titleInput${index}`); + const cardTitle = document.getElementById(`cardTitle${index}`); + if (input && cardTitle) { + const newTitle = input.value || `Camera ${index}`; + cardTitle.textContent = newTitle; + + if (event && event.target === input) { + currentSettings.camera_titles[index.toString()] = newTitle; + debouncedSaveSettings({ + camera_titles: { + [index.toString()]: newTitle + } + }); } - }); } - } } // Toggle titles function toggleTitles() { - const show = document.getElementById('toggleTitles').checked; - for (let i = 1; i <= 6; i++) { - const cardTitle = document.getElementById(`cardTitle${i}`); - if (cardTitle) { - cardTitle.style.display = show ? 'block' : 'none'; + const show = document.getElementById('toggleTitles').checked; + for (let i = 1; i <= 6; i++) { + const cardTitle = document.getElementById(`cardTitle${i}`); + if (cardTitle) { + cardTitle.style.display = show ? 'block' : 'none'; + } + } + + if (event && event.target === document.getElementById('toggleTitles')) { + currentSettings.display_options.show_titles = show; + debouncedSaveSettings({ + display_options: { + show_titles: show + } + }); } - } - - if (event && event.target === document.getElementById('toggleTitles')) { - currentSettings.display_options.show_titles = show; - debouncedSaveSettings({ - display_options: { - show_titles: show - } - }); - } } // Adjust title size function adjustTitleSize(size) { - document.getElementById('titleSizeValue').textContent = size; - for (let i = 1; i <= 6; i++) { - const cardTitle = document.getElementById(`cardTitle${i}`); - if (cardTitle) { - cardTitle.style.fontSize = `${size}rem`; + document.getElementById('titleSizeValue').textContent = size; + for (let i = 1; i <= 6; i++) { + const cardTitle = document.getElementById(`cardTitle${i}`); + if (cardTitle) { + cardTitle.style.fontSize = `${size}rem`; + } + } + + if (event && event.target === document.getElementById('titleSize')) { + currentSettings.display_options.title_size = parseFloat(size); + debouncedSaveSettings({ + display_options: { + title_size: parseFloat(size) + } + }); } - } - - if (event && event.target === document.getElementById('titleSize')) { - currentSettings.display_options.title_size = parseFloat(size); - debouncedSaveSettings({ - display_options: { - title_size: parseFloat(size) - } - }); - } } // Open camera fullscreen function openCameraFullscreen(cameraId) { - const customTitle = currentSettings.camera_titles[cameraId.toString()] || `Camera ${cameraId}`; - window.location.href = `/fullscreen/${cameraId}?title=${encodeURIComponent(customTitle)}`; + const customTitle = currentSettings.camera_titles[cameraId.toString()] || `Camera ${cameraId}`; + window.location.href = `/fullscreen/${cameraId}?title=${encodeURIComponent(customTitle)}`; } // Apply current settings function applyCurrentSettings() { - // Apply title visibility - const showTitles = currentSettings.display_options.show_titles; - if (document.getElementById('toggleTitles')) { - document.getElementById('toggleTitles').checked = showTitles; - } - - for (let i = 1; i <= 6; i++) { - const cardTitle = document.getElementById(`cardTitle${i}`); - if (cardTitle) { - cardTitle.style.display = showTitles ? 'block' : 'none'; - } - } - - // Apply title size - const titleSize = currentSettings.display_options.title_size; - if (document.getElementById('titleSizeValue')) { - document.getElementById('titleSizeValue').textContent = titleSize; - } - if (document.getElementById('titleSize')) { - document.getElementById('titleSize').value = titleSize; - } - - for (let i = 1; i <= 6; i++) { - const cardTitle = document.getElementById(`cardTitle${i}`); - if (cardTitle) { - cardTitle.style.fontSize = `${titleSize}rem`; - } - } - - // Apply card titles - for (let i = 1; i <= 6; i++) { - const cardTitle = document.getElementById(`cardTitle${i}`); - if (cardTitle) { - const title = currentSettings.camera_titles[i.toString()] || `Camera ${i}`; - cardTitle.textContent = title; + // Apply title visibility + const showTitles = currentSettings.display_options.show_titles; + const toggleTitlesEl = document.getElementById('toggleTitles'); + if (toggleTitlesEl) { + toggleTitlesEl.checked = showTitles; } - if (!tournamentActive) { - const input = document.getElementById(`titleInput${i}`); - if (input) { - input.value = currentSettings.camera_titles[i.toString()] || `Camera ${i}`; - } + for (let i = 1; i <= 6; i++) { + const cardTitle = document.getElementById(`cardTitle${i}`); + if (cardTitle) { + cardTitle.style.display = showTitles ? 'block' : 'none'; + } + } + + // Apply title size + const titleSize = currentSettings.display_options.title_size; + const titleSizeValueEl = document.getElementById('titleSizeValue'); + const titleSizeEl = document.getElementById('titleSize'); + + if (titleSizeValueEl) { + titleSizeValueEl.textContent = titleSize; + } + if (titleSizeEl) { + titleSizeEl.value = titleSize; + } + + for (let i = 1; i <= 6; i++) { + const cardTitle = document.getElementById(`cardTitle${i}`); + if (cardTitle) { + cardTitle.style.fontSize = `${titleSize}rem`; + } + } + + // Apply card titles + for (let i = 1; i <= 6; i++) { + const cardTitle = document.getElementById(`cardTitle${i}`); + if (cardTitle) { + const title = currentSettings.camera_titles[i.toString()] || `Camera ${i}`; + cardTitle.textContent = title; + } + + if (!tournamentActive) { + const input = document.getElementById(`titleInput${i}`); + if (input) { + input.value = currentSettings.camera_titles[i.toString()] || `Camera ${i}`; + } + } } - } } // Time and date updates function updateDateTime() { - const now = new Date(); - const hours = now.getHours().toString().padStart(2, '0'); - const minutes = now.getMinutes().toString().padStart(2, '0'); - const day = now.getDate().toString().padStart(2, '0'); - const month = (now.getMonth() + 1).toString().padStart(2, '0'); - const year = now.getFullYear(); + const now = new Date(); + const hours = now.getHours().toString().padStart(2, '0'); + const minutes = now.getMinutes().toString().padStart(2, '0'); + const day = now.getDate().toString().padStart(2, '0'); + const month = (now.getMonth() + 1).toString().padStart(2, '0'); + const year = now.getFullYear(); - const timeEl = document.getElementById('time'); - const dateEl = document.getElementById('date'); - - if (timeEl) timeEl.textContent = `${hours}:${minutes}`; - if (dateEl) dateEl.textContent = `${day}.${month}.${year}`; + const timeEl = document.getElementById('time'); + const dateEl = document.getElementById('date'); + + if (timeEl) timeEl.textContent = `${hours}:${minutes}`; + if (dateEl) dateEl.textContent = `${day}.${month}.${year}`; } - // Settings panel functionality - const menuButton = document.getElementById('menuButton'); - const settingsPanel = document.getElementById('settingsPanel'); - const overlay = document.getElementById('overlay'); - const closeBtn = document.getElementById('closeBtn'); - - function openSettings() { - settingsPanel.classList.add('active'); - overlay.classList.add('active'); + // Setup event listeners for menu + function setupEventListeners() { + console.log('Setting up event listeners...'); + + // Menu button click + if (menuButton) { + console.log('Adding menu button listener'); + menuButton.addEventListener('click', function(e) { + e.preventDefault(); + e.stopPropagation(); + console.log('Menu button clicked'); + openSettings(); + }); + } else { + console.error('Menu button not found!'); + } + + // Close button click + if (closeBtn) { + console.log('Adding close button listener'); + closeBtn.addEventListener('click', function(e) { + e.preventDefault(); + e.stopPropagation(); + console.log('Close button clicked'); + closeSettings(); + }); + } + + // Overlay click + if (overlay) { + console.log('Adding overlay listener'); + overlay.addEventListener('click', function(e) { + if (e.target === overlay) { + console.log('Overlay clicked'); + closeSettings(); + } + }); + } } - function closeSettings() { - settingsPanel.classList.remove('active'); - overlay.classList.remove('active'); - } - - // Event listeners - if (menuButton) menuButton.addEventListener('click', openSettings); - if (closeBtn) closeBtn.addEventListener('click', closeSettings); - if (overlay) overlay.addEventListener('click', closeSettings); - // Keyboard shortcuts - document.addEventListener('keydown', function(event) { - if (event.key === 'Escape' && settingsPanel.classList.contains('active')) { - closeSettings(); - } else if (tournamentActive) { - if (event.key === 'ArrowLeft') { - event.preventDefault(); - changeRound(-1); - } else if (event.key === 'ArrowRight') { - event.preventDefault(); - changeRound(1); - } - } - }); - - // Initialize - document.addEventListener('DOMContentLoaded', function() { - updateDateTime(); - setInterval(updateDateTime, 1000); - updateRoundNavigation(); - applyCurrentSettings(); - addTouchFeedback(); - }); + function setupKeyboardShortcuts() { + document.addEventListener('keydown', function(event) { + if (event.key === 'Escape' && settingsPanel && settingsPanel.classList.contains('active')) { + console.log('Escape pressed, closing settings'); + closeSettings(); + } else if (tournamentActive) { + if (event.key === 'ArrowLeft') { + event.preventDefault(); + changeRound(-1); + } else if (event.key === 'ArrowRight') { + event.preventDefault(); + changeRound(1); + } + } + + // Remote control shortcuts + if (remoteControlled) { + if (event.key === 'r' || event.key === 'R') { + if (event.ctrlKey || event.metaKey) { + // Allow normal refresh + return; + } + event.preventDefault(); + window.location.reload(); + } + } + }); + } // Add touch feedback for interactive elements on touch devices function addTouchFeedback() { - if (isMobileDevice()) { - const touchElements = document.querySelectorAll('.camera-card, .nav-btn'); - - touchElements.forEach(element => { - element.addEventListener('touchstart', function(e) { - if (!this.disabled) { - this.style.transform = 'scale(0.95) translateY(0)'; - this.style.transition = 'transform 0.1s ease'; - } - }); - - element.addEventListener('touchend', function(e) { - if (!this.disabled) { - setTimeout(() => { - this.style.transform = ''; - this.style.transition = 'transform 0.2s ease'; - }, 100); - } - }); - - element.addEventListener('touchcancel', function(e) { - if (!this.disabled) { - this.style.transform = ''; - this.style.transition = 'transform 0.2s ease'; - } - }); - }); - } + if (isMobileDevice()) { + const touchElements = document.querySelectorAll('.camera-card, .nav-btn, .hamburger-menu'); + + touchElements.forEach(element => { + element.addEventListener('touchstart', function(e) { + if (!this.disabled) { + this.style.transform = 'scale(0.95) translateY(0)'; + this.style.transition = 'transform 0.1s ease'; + } + }); + + element.addEventListener('touchend', function(e) { + if (!this.disabled) { + setTimeout(() => { + this.style.transform = ''; + this.style.transition = 'transform 0.2s ease'; + }, 100); + } + }); + + element.addEventListener('touchcancel', function(e) { + if (!this.disabled) { + this.style.transform = ''; + this.style.transition = 'transform 0.2s ease'; + } + }); + }); + } } - + + // Initialize everything + document.addEventListener('DOMContentLoaded', function() { + + // Initialize DOM elements first + initializeDOMElements(); + + // Setup event listeners + setupEventListeners(); + + // Setup keyboard shortcuts + setupKeyboardShortcuts(); + + // Initialize other features + updateDateTime(); + setInterval(updateDateTime, 1000); + updateRoundNavigation(); + applyCurrentSettings(); + addTouchFeedback(); + }); + + // Cleanup on page unload + window.addEventListener('beforeunload', function() { + if (remotePollingInterval) { + clearInterval(remotePollingInterval); + } + }); + + + + + \ No newline at end of file diff --git a/templates/league_scoreboard_display.html b/templates/league_scoreboard_display.html index 54aaec9..b965c7c 100644 --- a/templates/league_scoreboard_display.html +++ b/templates/league_scoreboard_display.html @@ -2,7 +2,7 @@ - League Results + League Results - - -
- - -
- -
- {% if tournament %} -
-
🎯 Shooting Tournament
-
- {{ tournament.total_players }} players • {{ tournament.total_rounds }} rounds - {% if tournament.current_round %} - • Currently on Round {{ tournament.current_round }} - {% endif %} -
-
- -
- {% for round in tournament.rounds %} - {% set is_current = tournament.current_round == round.round_number %} - {% set is_completed = tournament.current_round > round.round_number %} -
-
-
Round {{ round.round_number }}
- {% if is_current %} -
Current
- {% elif is_completed %} -
Completed
- {% else %} -
Waiting
- {% endif %} -
- -
- {% for position in range(1, 7) %} - {% set player = round.players[position-1] if position <= round.players|length else none %} -
-
{{ position }}
- {% if player %} -
{{ player.name }}
-
ID: {{ player.id }}
- {% else %} -
Empty
- {% endif %} -
- {% endfor %} -
-
- {% endfor %} -
- - -
-
📊 Tournament Summary
-
-
- Total Players: - {{ tournament.total_players }} -
-
- Total Rounds: - {{ tournament.total_rounds }} -
-
- Current Round: - {{ tournament.current_round or 'Not set' }} -
-
- Updates: - On Round Change -
-
- Created: - {{ tournament.created_at[:10] if tournament.created_at else 'Unknown' }} -
-
- Status: - Active -
-
-
- {% else %} - -
-

📋 No Active Tournament

-

No tournament is currently running. Check the Results tab for completed tournaments or use the dashboard to start a new tournament.

-
- {% endif %} -
- - -
- 🆕 New Round Starting... -
- - - - \ No newline at end of file diff --git a/templates/mobile_league_results.html b/templates/mobile_league_results.html deleted file mode 100644 index 199e1c8..0000000 --- a/templates/mobile_league_results.html +++ /dev/null @@ -1,934 +0,0 @@ - - - - - 📱 League Results - - - - -
- - -
- -
-
-
🏆 League Championship
-
- Final Results - Best 5 of 6 Tournaments -
-
- {% if league.tournament_type == '40_targets' %} - 40 Targets Format - {% else %} - 20 Targets Format - {% endif %} -
-
- -
-
-
{{ participants|length if participants else 0 }}
-
Participants
-
-
-
{{ league.total_tournaments }}
-
Tournaments
-
-
-
{% if participants and participants|length > 0 %}{{ participants[0].final_score }}{% else %}0{% endif %}
-
Highest Score
-
-
-
{{ league.finished_at[:10] if league.finished_at else 'Today' }}
-
Completed
-
-
- - {% if participants and participants|length > 0 %} -
-
🎉 League Complete!
-
- Congratulations to {{ participants[0].name }} for winning the league! -
-
- -
- {% for participant in participants %} -
-
-
{{ participant.rank }}
-
- {% if participant.rank == 1 %}st - {% elif participant.rank == 2 %}nd - {% elif participant.rank == 3 %}rd - {% else %}th - {% endif %} -
- {% if participant.rank == 1 %} -
🥇
- {% elif participant.rank == 2 %} -
🥈
- {% elif participant.rank == 3 %} -
🥉
- {% endif %} -
- -
-
{{ participant.name }}
-
-
ID: {{ participant.id }}
-
-
- -
-
{{ participant.final_score }}
-
Total: {{ participant.total_score }}
-
Final Score
-
-
- {% endfor %} -
- - - - {% set max_breakdowns = 3 %} - {% if participants|length < 3 %} - {% set max_breakdowns = participants|length %} - {% endif %} - - {% for i in range(max_breakdowns) %} - {% set participant = participants[i] %} -
-
- {% if participant.rank == 1 %}🥇{% elif participant.rank == 2 %}🥈{% elif participant.rank == 3 %}🥉{% endif %} - {{ participant.name }}'s Tournament History -
-
- {% for tournament_num in range(1, 7) %} - {% set result = participant.tournament_results - | selectattr("tournament", "equalto", tournament_num) - | list - | first %} - - {% if result %} - {% if (result.joker is defined and result.joker) or (result.participated is defined and not result.participated) %} -
-
T{{ tournament_num }}
-
🃏 Joker
-
- {% else %} - {% set all_participated_scores = participant.tournament_results - | selectattr("participated", "defined") - | selectattr("participated") - | selectattr("score", "defined") - | map(attribute="score") - | list %} - {% set sorted_scores = all_participated_scores | sort(reverse=true) %} - {% set is_excluded = sorted_scores | length > 5 and result.score == sorted_scores[-1] %} - - {% if is_excluded %} -
-
T{{ tournament_num }}
-
{{ result.score }} ❌
-
- {% else %} -
-
T{{ tournament_num }}
-
{{ result.score }} ✅
-
- {% endif %} - {% endif %} - {% else %} -
-
T{{ tournament_num }}
-
-
-
- {% endif %} - {% endfor %} - -
-
- {% endfor %} - - -
-
📖 How Final Scores Are Calculated
-
-
- 🏆 League Rule: Best 5 out of 6 tournaments count toward final ranking -
-
- 🃏 Joker System: Each player can skip 1 tournament without penalty -
-
-
- - Counted toward final score -
-
- - Excluded (worst score) -
-
- 🃏 - Joker used (skipped) -
-
-
-
- {% else %} -
-
-

No League Results Available

-

League results will appear here when the league is complete.

-
-
- {% endif %} -
- - -
- - - - \ No newline at end of file diff --git a/templates/mobile_menu.html b/templates/mobile_menu.html deleted file mode 100644 index acf3382..0000000 --- a/templates/mobile_menu.html +++ /dev/null @@ -1,382 +0,0 @@ - - - - - 📱 Camera Dashboard - - - - -
- -
Camera Dashboard
-
Loading mobile interface...
- -
-
Detecting tournament state...
- - - - - - -
- - - - \ No newline at end of file diff --git a/templates/mobile_results.html b/templates/mobile_results.html deleted file mode 100644 index 96d0b85..0000000 --- a/templates/mobile_results.html +++ /dev/null @@ -1,696 +0,0 @@ - - - - - 📱 Tournament Results - - - - -
- - -
- -
- - {% if participants %} - {% if participants|length > 0 %} -
-
🎉 Tournament Complete!
-
- Congratulations to {{ participants[0].name }} for winning! -
-
- {% endif %} - -
- {% for participant in participants %} -
-
-
{{ participant.rank }}
-
- {% if participant.rank == 1 %}st - {% elif participant.rank == 2 %}nd - {% elif participant.rank == 3 %}rd - {% else %}th - {% endif %} -
- {% if participant.rank == 1 %} -
🥇
- {% elif participant.rank == 2 %} -
🥈
- {% elif participant.rank == 3 %} -
🥉
- {% endif %} -
- -
-
{{ participant.name }}
-
ID: {{ participant.id }}
-
- -
-
{{ participant.total_score }}
-
Points
-
-
- {% endfor %} -
- {% else %} -
-
-

No Results Available

-

Tournament results will appear here when scoring is complete.

-
-
- {% endif %} - - -
-
📊 Tournament Information
-
-
- Total Participants: - {{ participants|length if participants else 0 }} -
-
- Tournament ID: - {{ results.tournament_id[:10] if results.tournament_id else 'Unknown' }}... -
-
- Created: - {{ results.created_at[:10] if results.created_at else 'Unknown' }} -
-
- Status: - Completed -
- {% if participants and participants|length > 0 %} -
- Highest Score: - {{ participants[0].total_score }} points -
-
- Winner: - {{ participants[0].name }} -
- {% endif %} -
-
-
- - -
- - - - \ No newline at end of file diff --git a/templates/mobile_streams.html b/templates/mobile_streams.html index 6da9457..b26a1f8 100644 --- a/templates/mobile_streams.html +++ b/templates/mobile_streams.html @@ -2,7 +2,7 @@ - 📱 Camera Streams + 📱 Camera Streams @@ -544,62 +657,65 @@
- -
- -
-
📊 Statistics
-
-
-
{{ stats.total_tournaments }}
-
Tournaments
-
-
-
{{ stats.total_leagues }}
-
Leagues
-
-
-
{{ stats.best_tournament_score }}
-
Best Score
-
-
-
{{ stats.average_tournament_score|round|int if stats.average_tournament_score > 0 else 0 }}
-
Average
-
-
-
{{ stats.total_shots_fired|default(0) }}
-
Total Shots
-
-
-
{{ stats.worst_tournament_score if stats.worst_tournament_score > 0 else 0 }}
-
Worst Score
-
-
+ +
+
+ 🏆 +
{{ stats.total_tournaments }}
+
Tournaments
+
+ 🎖️ +
{{ stats.total_leagues }}
+
Leagues
+
+
+ +
{{ stats.best_tournament_score }}
+
Best Score
+
+
+ 📊 +
{{ stats.average_tournament_score|round|int if stats.average_tournament_score > 0 else 0 }}
+
Average
+
+
+ 🔫 +
{{ stats.total_shots_fired|default(0) }}
+
Total Shots
+
+
+ 📉 +
{{ stats.worst_tournament_score if stats.worst_tournament_score > 0 else 0 }}
+
Worst Score
+
+
- + +
-
📈 Performance by Tournament Type
- +
📈 Performance by Tournament Type
+
@@ -608,15 +724,15 @@
0
-
Games
+
Games
0
-
Average
+
Average
0
-
Best
+
Best
@@ -680,14 +796,14 @@
-
🎯 Tournament History
+
🎯 Tournament History
{% if stats.tournament_history %}
{% for tournament in stats.tournament_history[:10] %}
-
{{ tournament.date[:10] if tournament.date != 'Unknown' else 'Unknown Date' }}
-
{{ tournament.tournament_type.replace('_', ' ')|title }} • {{ tournament.shots_fired }} shots
+
{{ tournament.date[:10] if tournament.date != 'Unknown' else (translations.analysis.unknown_date if translations else 'Unknown Date') }}
+
{{ tournament.tournament_type.replace('_', ' ')|title }} • {{ tournament.shots_fired }} {{ translations.results.shots if translations else 'shots' }}
{{ tournament.score }}
@@ -718,7 +834,7 @@
{{ league.date[:10] if league.date != 'Unknown' else 'Unknown Date' }}
League Championship • {{ league.tournaments_participated }}/6 tournaments - {% if league.joker_used %} • 🃏 Joker Used{% endif %} + {% if league.joker_used %} • 🃏 {{ translations.league.joker_used if translations else 'Joker Used' }}{% endif %}
{{ league.final_score }}
@@ -732,7 +848,7 @@
{% for result in league.tournament_results %} - T{{ result.tournament }}: {{ result.score if result.participated else 'Joker' }} + T{{ result.tournament }}: {{ result.score if result.participated else (translations.league.joker_used if translations else 'Joker') }} {% endfor %}
@@ -1122,5 +1238,20 @@ // Initialize page when DOM is loaded document.addEventListener('DOMContentLoaded', initializePage); + + + + \ No newline at end of file diff --git a/templates/results_calculator.html b/templates/results_calculator.html index c0b10e0..0b32666 100644 --- a/templates/results_calculator.html +++ b/templates/results_calculator.html @@ -2,7 +2,7 @@ - Results Calculator + Results Calculator + +
-
+
-
Tournament Results
-
Final Rankings & Scores
+
+ {% if results and results.tournament_type == '4_targets' %} + 🎯 Tournament Results + {% elif results and results.tournament_type == '40_targets' %} + 💪 Tournament Results + {% else %} + ⚡ Tournament Results + {% endif %} +
+
Final Rankings & Scores
- {{ participants|length }} - Participants + {{ participants|length }} + Participants
- {{ (participants|length * 40) if participants else 0 }} - Total Shots + 0 + Total Shots
- {% if participants and participants|length > 0 %}{{ participants[0].total_score }}{% else %}0{% endif %} - Highest Score + 0 + Highest Score +
+
+ 0 + Most 10s
- {% if participants and participants|length >= 3 %} -
-
🏆 Top 3 Winners
-
- {% for i in range(3) %} - {% set participant = participants[i] %} -
-
-
{{ participant.rank }}
-
- {% if participant.rank == 1 %}st - {% elif participant.rank == 2 %}nd - {% elif participant.rank == 3 %}rd - {% else %}th - {% endif %} -
-
- {% if participant.rank == 1 %}🥇 - {% elif participant.rank == 2 %}🥈 - {% elif participant.rank == 3 %}🥉 - {% endif %} -
-
- -
-
{{ participant.name }}
-
ID: {{ participant.id }}
-
- -
-
{{ participant.total_score }}
-
Points
-
-
- {% endfor %} +
+
🏆 Top 3 Winners
+
+
- {% endif %}
-

📊 Complete Rankings

+

📊 Complete Rankings

+ +
- - - - - + + + + - - {% for participant in participants %} - - - - - - - - {% endfor %} + +
RankParticipantIDScoreStatusRankParticipantScore10s
- {{ participant.rank }} - {% if participant.rank == 1 %}🥇 - {% elif participant.rank == 2 %}🥈 - {% elif participant.rank == 3 %}🥉 - {% endif %} - {{ participant.name }}{{ participant.id }}{{ participant.total_score }} - - ✓ - -
- // Initialize TV mode + + + \ No newline at end of file diff --git a/templates/tournament.html b/templates/tournament.html index 888e5b9..6c32b28 100644 --- a/templates/tournament.html +++ b/templates/tournament.html @@ -2,7 +2,7 @@ - Tournament Management + Tournament Management