"""
Grading system views for evaluating model answers.

Implements blind grading with aliases, multi-grader support, and progressive disclosure
of model identities only after completing all tiers.
"""

from django.views.generic import DetailView, UpdateView, View, TemplateView
from django.contrib.auth.mixins import LoginRequiredMixin
from django.shortcuts import redirect, get_object_or_404
from django.contrib import messages
from django.http import JsonResponse
from django.db import transaction, models
from django.utils import timezone
from django.urls import reverse
import random
import string

from .models import Question
from model_evaluation.models import (
    ModelAnswer, ModelGradingSession, ModelGradingAlias,
    ModelGrading, GradingState, Model
)
from review_invitations.models import ReviewAssignment
from accounts.permissions import OwnerOrAdminMixin


class GradingAccessMixin:
    """Mixin to check if user can grade a question."""
    
    def dispatch(self, request, *args, **kwargs):
        """Check if user has permission to grade this question."""
        self.question = get_object_or_404(Question, pk=kwargs.get('question_id'))
        
        # Check if there are released answers
        self.released_answers = ModelAnswer.objects.filter(
            question=self.question,
            released_for_grading=True
        ).select_related('model__tier', 'model__company')
        
        if not self.released_answers.exists():
            messages.error(request, "No model answers have been released for grading yet.")
            return redirect('questions:detail', pk=self.question.pk)
        
        # Check if user can grade (author, admin, or assigned reviewer)
        if hasattr(request.user, 'participant'):
            is_author = self.question.author == request.user.participant
            is_admin = request.user.participant.is_admin
            has_review_assignment = ReviewAssignment.objects.filter(
                user=request.user,
                question=self.question,
                is_active=True
            ).exists()

            if not (is_author or is_admin or has_review_assignment):
                messages.error(request, "You don't have permission to grade this question.")
                return redirect('questions:detail', pk=self.question.pk)
        else:
            messages.error(request, "You need to be logged in with a participant account to grade.")
            return redirect('questions:detail', pk=self.question.pk)
            
        return super().dispatch(request, *args, **kwargs)


class GradingOverviewView(LoginRequiredMixin, GradingAccessMixin, DetailView):
    """
    Display grading progress overview for a question.

    Shows all model answers with their grading status, manages aliases,
    and provides navigation to individual grading interfaces.
    Supports admin inspection mode to view other people's sessions.
    """
    template_name = 'questions/grading_overview.html'
    context_object_name = 'question'

    def dispatch(self, request, *args, **kwargs):
        """Check permissions and handle admin inspection mode."""
        # Check if this is admin inspection mode
        session_id = request.GET.get('inspect_session')
        self.admin_inspection_mode = False
        self.inspected_session = None

        if session_id:
            # Admin trying to inspect a specific session
            if hasattr(request.user, 'participant') and request.user.participant.is_admin:
                try:
                    self.inspected_session = ModelGradingSession.objects.get(pk=session_id)
                    self.question = self.inspected_session.question
                    self.admin_inspection_mode = True
                    # Override the regular access check since admin is inspecting
                    self.released_answers = ModelAnswer.objects.filter(
                        question=self.question,
                        released_for_grading=True
                    ).select_related('model__tier', 'model__company')
                    return super(GradingAccessMixin, self).dispatch(request, *args, **kwargs)
                except ModelGradingSession.DoesNotExist:
                    messages.error(request, "Session not found.")
                    return redirect('questions:grading_inspection')
            else:
                messages.error(request, "You don't have permission to inspect sessions.")
                return redirect('home')

        # Regular grading access
        return super().dispatch(request, *args, **kwargs)

    def get_object(self):
        return self.question

    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)

        # If in admin inspection mode, use the inspected session
        if self.admin_inspection_mode and self.inspected_session:
            session = self.inspected_session
            context['admin_inspection_mode'] = True
            context['inspected_grader'] = session.grader

            # Build session data for the inspected session
            session_data = self._build_session_data(session)
            context['sessions_data'] = [session_data]
            context['session'] = session
        else:
            # Get ALL existing sessions for this grader and question
            all_sessions = ModelGradingSession.objects.filter(
                grader=self.request.user,
                question=self.question
            ).order_by('created_at')

            # Get all answer IDs that have aliases in any session
            all_aliased_answer_ids = set(
                ModelGradingAlias.objects.filter(
                    session__in=all_sessions
                ).values_list('model_answer_id', flat=True)
            )

            # Get currently released answer IDs
            current_released_ids = set(
                self.released_answers.values_list('id', flat=True)
            )

            # Check if there are new answers that need a new session
            new_answer_ids = current_released_ids - all_aliased_answer_ids

            if new_answer_ids:
                # Create new session for new answers
                new_session = ModelGradingSession.objects.create(
                    grader=self.request.user,
                    question=self.question,
                    session_status='active'
                )

                # Generate aliases for new answers, continuing from existing max
                new_answers = self.released_answers.filter(id__in=new_answer_ids)
                self._generate_aliases_for_session(new_session, new_answers)

                messages.info(
                    self.request,
                    f"New grading session created with {len(new_answer_ids)} answer(s). "
                    f"Aliases continue from previous sessions."
                )

                # Refresh sessions queryset to include new session
                all_sessions = ModelGradingSession.objects.filter(
                    grader=self.request.user,
                    question=self.question
                ).order_by('created_at')
            elif not all_sessions.exists():
                # No sessions at all, create initial one
                new_session = ModelGradingSession.objects.create(
                    grader=self.request.user,
                    question=self.question,
                    session_status='active'
                )
                self._generate_aliases_for_session(new_session, self.released_answers)
                messages.info(self.request, "Grading session initialized with randomized aliases.")

                # Set sessions queryset
                all_sessions = ModelGradingSession.objects.filter(id=new_session.id)

            # Pass all sessions to template (will be displayed as separate panels)
            context['sessions'] = all_sessions

            # Build session data for template
            sessions_data = []
            for session in all_sessions:
                session_data = self._build_session_data(session)
                sessions_data.append(session_data)

            context['sessions_data'] = sessions_data

            # For backwards compatibility, if there's only one session, also set it as 'session'
            if all_sessions.count() == 1:
                context['session'] = all_sessions.first()

        return context

    def _build_session_data(self, session):
        """Build complete data for a single session."""
        # Get all aliases for this session
        aliases = ModelGradingAlias.objects.filter(
            session=session
        ).select_related('model_answer__model__tier', 'model_answer__model__company')

        # Get all gradings for this session
        gradings = ModelGrading.objects.filter(
            session=session
        ).select_related('model_answer')

        # Create a mapping of model_answer_id to grading
        grading_map = {g.model_answer_id: g for g in gradings}

        # Organize aliases by tier group
        tier1_aliases = []
        tier23_aliases = []
        tier4plus_aliases = []

        for alias in aliases:
            # Get or create grading for this alias
            grading = grading_map.get(alias.model_answer_id)
            if not grading:
                # Create grading record if it doesn't exist
                grading = ModelGrading.objects.create(
                    session=session,
                    model_answer=alias.model_answer,
                    grader=self.request.user,
                    grading_status='not_started'
                )
            
            # Helper function to convert GradingState to display value
            def get_state_value(state_obj):
                if state_obj is None:
                    return None
                if state_obj.state_code == 'true':
                    return True
                elif state_obj.state_code == 'false':
                    return False
                elif state_obj.state_code == 'not_sure':
                    return None  # Will show as ?
                elif state_obj.state_code == 'not_applicable':
                    return 'N/A'
                return None
            
            # Add grading info to alias
            alias_data = {
                'alias': alias.alias,
                'alias_order': alias.alias_order,
                'model_answer_id': alias.model_answer_id,
                'grading_status': grading.grading_status,
                'is_complete': grading.is_complete,
                'progress_grade': grading.get_progress_grade_display() if grading.progress_grade is not None else None,
                'progress_grade_raw': grading.progress_grade,
                # Add all grading fields for table display (converted to boolean/None)
                'error_incorrect_logic': get_state_value(grading.error_incorrect_logic),
                'error_hallucinated': get_state_value(grading.error_hallucinated),
                'error_calculation': get_state_value(grading.error_calculation),
                'error_conceptual': get_state_value(grading.error_conceptual),
                'achievement_understanding': get_state_value(grading.achievement_understanding),
                'achievement_correct_result': get_state_value(grading.achievement_correct_result),
                'achievement_insight': get_state_value(grading.achievement_insight),
                'achievement_usefulness': get_state_value(grading.achievement_usefulness),
            }
            
            # If session is finalized, include model identity
            if session.is_finalized:
                alias_data['model_name'] = alias.model_answer.model.model_name
                alias_data['company_name'] = alias.model_answer.model.company.company_name
            
            if alias.tier_group == 1:
                tier1_aliases.append(alias_data)
            elif alias.tier_group == 2:
                tier23_aliases.append(alias_data)
            else:  # tier_group == 3 (Tier 4+)
                tier4plus_aliases.append(alias_data)
        
        # Sort by alias order for consistent display
        tier1_aliases.sort(key=lambda x: x['alias_order'])
        tier23_aliases.sort(key=lambda x: x['alias_order'])
        tier4plus_aliases.sort(key=lambda x: x['alias_order'])

        # Calculate completion statistics
        total_count = len(aliases)
        completed_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'completed')
        in_progress_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'in_progress')
        given_up_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'given_up')

        # Return all session data as dictionary
        return {
            'session': session,
            'tier1_aliases': tier1_aliases,
            'tier23_aliases': tier23_aliases,
            'tier4plus_aliases': tier4plus_aliases,
            'total_count': total_count,
            'completed_count': completed_count,
            'in_progress_count': in_progress_count,
            'given_up_count': given_up_count,
            'not_started_count': total_count - completed_count - in_progress_count - given_up_count,
            'all_complete': (completed_count + given_up_count) == total_count and total_count > 0,
        }
    
    def _generate_aliases_for_session(self, session, answers):
        """
        Generate randomized aliases for a specific set of answers.

        Continues alphabetical numbering from any previous sessions by the same grader
        for the same question.
        """
        # Get max alias_order from ALL sessions for this grader/question (not just current)
        max_order = ModelGradingAlias.objects.filter(
            session__grader=self.request.user,
            session__question=self.question
        ).aggregate(models.Max('alias_order'))['alias_order__max']

        if max_order is None:
            max_order = -1  # Start at 0 if no previous aliases exist

        # Sort answers by tier, then shuffle within each tier
        answers_list = list(answers.order_by('model__tier__tier_number', 'id'))

        # Group answers by tier
        tier_groups = {}
        for answer in answers_list:
            tier_num = answer.model.tier.tier_number
            if tier_num not in tier_groups:
                tier_groups[tier_num] = []
            tier_groups[tier_num].append(answer)

        # Shuffle within each tier group
        for tier_answers in tier_groups.values():
            random.shuffle(tier_answers)

        # Combine in order by tier number
        all_answers = []
        for tier_num in sorted(tier_groups.keys()):
            all_answers.extend(tier_groups[tier_num])

        # Generate aliases starting from max_order + 1
        with transaction.atomic():
            for i, answer in enumerate(all_answers):
                alias_order = max_order + 1 + i

                # Generate alias letter (A, B, C, ..., Z, AA, AB, ...)
                if alias_order < 26:
                    alias = f"Answer {chr(65 + alias_order)}"
                else:
                    # For more than 26 answers, use AA, AB, AC, etc.
                    first_letter = chr(65 + (alias_order // 26) - 1)
                    second_letter = chr(65 + (alias_order % 26))
                    alias = f"Answer {first_letter}{second_letter}"

                # Map tier numbers to tier groups
                # Tier 1 → group 1
                # Tiers 2-3 → group 2
                # Tier 4+ → group 3
                tier_num = answer.model.tier.tier_number
                if tier_num == 1:
                    tier_group = 1
                elif tier_num in [2, 3]:
                    tier_group = 2
                else:  # Tier 4 and beyond
                    tier_group = 3

                ModelGradingAlias.objects.create(
                    session=session,
                    model_answer=answer,
                    alias=alias,
                    alias_order=alias_order,
                    tier_group=tier_group
                )


class GradingDetailView(LoginRequiredMixin, GradingAccessMixin, UpdateView):
    """
    Main grading interface for a specific model answer.

    Displays the AI solution, model solution side-by-side with grading controls.
    Supports admin inspection mode for read-only viewing.
    """
    template_name = 'questions/grading_detail.html'
    model = ModelGrading
    fields = []  # We'll handle form processing manually

    def dispatch(self, request, *args, **kwargs):
        """Check permissions and handle admin inspection mode."""
        # Check if this is admin inspection mode
        session_id = request.GET.get('inspect_session')
        self.admin_inspection_mode = False
        self.inspected_session = None

        if session_id:
            # Admin trying to inspect a specific session
            if hasattr(request.user, 'participant') and request.user.participant.is_admin:
                try:
                    self.inspected_session = ModelGradingSession.objects.get(pk=session_id)
                    self.question = self.inspected_session.question
                    self.admin_inspection_mode = True
                    # Override the regular access check since admin is inspecting
                    self.released_answers = ModelAnswer.objects.filter(
                        question=self.question,
                        released_for_grading=True
                    ).select_related('model__tier', 'model__company')
                    return super(GradingAccessMixin, self).dispatch(request, *args, **kwargs)
                except ModelGradingSession.DoesNotExist:
                    messages.error(request, "Session not found.")
                    return redirect('questions:grading_inspection')
            else:
                messages.error(request, "You don't have permission to inspect sessions.")
                return redirect('home')

        # Regular grading access
        return super().dispatch(request, *args, **kwargs)

    def get_object(self):
        """Get or create the grading object for this answer."""
        alias = self.kwargs.get('alias')

        # Get the alias mapping first (alias is unique across all sessions for this user)
        # In admin inspection mode, look up via inspected session
        if self.admin_inspection_mode and self.inspected_session:
            alias_obj = get_object_or_404(
                ModelGradingAlias,
                session=self.inspected_session,
                alias=alias
            )
        else:
            alias_obj = get_object_or_404(
                ModelGradingAlias,
                session__grader=self.request.user,
                session__question=self.question,
                alias=alias
            )

        # Get the session from the alias
        session = alias_obj.session

        # In admin inspection mode, just get the existing grading (don't create)
        if self.admin_inspection_mode:
            try:
                grading = ModelGrading.objects.get(
                    session=session,
                    model_answer=alias_obj.model_answer
                )
            except ModelGrading.DoesNotExist:
                # If no grading exists yet, create a placeholder with not_started status
                # but use the session's grader, not the admin
                grading = ModelGrading.objects.create(
                    session=session,
                    model_answer=alias_obj.model_answer,
                    grader=session.grader,  # Use the original grader, not the admin
                    grading_status='not_started'
                )
        else:
            # Regular mode - get or create the grading
            grading, created = ModelGrading.objects.get_or_create(
                session=session,
                model_answer=alias_obj.model_answer,
                grader=self.request.user,
                defaults={'grading_status': 'in_progress'}
            )

            # If just created or was not_started, mark as in_progress
            if created or grading.grading_status == 'not_started':
                grading.grading_status = 'in_progress'
                grading.save()
        
        # Store alias object for use in context
        self.alias_obj = alias_obj
        self.session = session
        
        return grading
    
    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        context['question'] = self.question
        context['alias'] = self.kwargs.get('alias')
        context['alias_obj'] = self.alias_obj
        context['session'] = self.session
        context['model_answer'] = self.object.model_answer

        # Add admin inspection mode context
        if self.admin_inspection_mode:
            context['admin_inspection_mode'] = True
            context['inspected_grader'] = self.session.grader
        
        # Get all aliases for tab navigation
        all_aliases = ModelGradingAlias.objects.filter(
            session=self.session
        ).select_related('model_answer').order_by('alias_order')
        
        # Get grading status for each alias
        gradings = ModelGrading.objects.filter(
            session=self.session
        ).values('model_answer_id', 'grading_status')
        
        grading_map = {g['model_answer_id']: g['grading_status'] for g in gradings}
        
        # Build tab data
        tabs = []
        for alias in all_aliases:
            status = grading_map.get(alias.model_answer_id, 'not_started')
            url = reverse('questions:grading_detail', kwargs={
                'question_id': self.question.pk,
                'alias': alias.alias
            })
            # Preserve inspect_session parameter if in admin inspection mode
            if self.admin_inspection_mode and self.request.GET.get('inspect_session'):
                url += f"?inspect_session={self.request.GET.get('inspect_session')}"
            tabs.append({
                'alias': alias.alias,
                'tier_group': alias.tier_group,
                'status': status,
                'is_current': alias.alias == self.kwargs.get('alias'),
                'url': url
            })
        
        context['tabs'] = tabs
        
        # Get grading states for form
        context['grading_states'] = GradingState.objects.filter(is_active=True)
        
        # Get current grading values
        context['grading'] = self.object
        
        return context
    
    def _get_next_incomplete_url(self):
        """Find the next incomplete answer and return its URL."""
        # Get all aliases ordered
        all_aliases = ModelGradingAlias.objects.filter(
            session=self.session
        ).select_related('model_answer').order_by('alias_order')
        
        # Get all gradings
        gradings = ModelGrading.objects.filter(
            session=self.session
        ).select_related('model_answer')
        
        grading_map = {g.model_answer_id: g for g in gradings}
        
        # Find the current alias index
        current_index = None
        incomplete_aliases = []
        
        for i, alias in enumerate(all_aliases):
            if alias.model_answer_id == self.object.model_answer_id:
                current_index = i
            
            # Check if this alias is incomplete
            grading = grading_map.get(alias.model_answer_id)
            if not grading or grading.grading_status not in ['completed', 'given_up']:
                incomplete_aliases.append((i, alias))
        
        if current_index is not None and incomplete_aliases:
            # Find the next incomplete after current
            for idx, alias in incomplete_aliases:
                if idx > current_index:
                    return reverse('questions:grading_detail', kwargs={
                        'question_id': self.question.pk,
                        'alias': alias.alias
                    })
            
            # If no incomplete after current, try from beginning
            for idx, alias in incomplete_aliases:
                if idx != current_index:
                    return reverse('questions:grading_detail', kwargs={
                        'question_id': self.question.pk,
                        'alias': alias.alias
                    })
        
        # If all are complete or no next found, return overview URL
        return reverse('questions:grading_overview', kwargs={'question_id': self.question.pk})
    
    def post(self, request, *args, **kwargs):
        """Handle AJAX auto-save requests."""
        # Prevent any modifications in admin inspection mode
        if self.admin_inspection_mode:
            return JsonResponse({'success': False, 'error': 'Read-only mode'}, status=403)

        self.object = self.get_object()

        if request.headers.get('X-Requested-With') == 'XMLHttpRequest':
            # AJAX auto-save
            try:
                # Update grading fields
                data = request.POST
                
                # Update error indicators
                for field in ['error_incorrect_logic', 'error_hallucinated', 
                             'error_calculation', 'error_conceptual']:
                    value = data.get(field)
                    if value:
                        setattr(self.object, f'{field}_id', int(value) if value != 'null' else None)
                
                # Update achievement indicators  
                for field in ['achievement_understanding', 'achievement_correct_result',
                             'achievement_insight', 'achievement_usefulness']:
                    value = data.get(field)
                    if value:
                        setattr(self.object, f'{field}_id', int(value) if value != 'null' else None)
                
                # Update progress grade
                progress = data.get('progress_grade')
                if progress is not None:
                    self.object.progress_grade = int(progress) if progress != '' else None
                
                # Update text fields
                # Only update comments if provided (to preserve when marking incomplete from completed state)
                if 'comments' in data:
                    self.object.comments = data.get('comments', '')
                
                # Only update flag if provided (to preserve when marking incomplete from completed state)
                if 'flag_for_organizers' in data:
                    self.object.flag_for_organizers = data.get('flag_for_organizers') == 'true'
                
                # Update status based on completion
                next_url = None
                if data.get('mark_complete') == 'true':
                    self.object.grading_status = 'completed'
                    # Find next incomplete answer
                    next_url = self._get_next_incomplete_url()
                elif data.get('give_up') == 'true':
                    self.object.grading_status = 'given_up'
                elif data.get('mark_incomplete') == 'true':
                    self.object.grading_status = 'in_progress'
                
                self.object.save()
                
                response_data = {
                    'success': True,
                    'is_complete': self.object.is_complete,
                    'grading_status': self.object.grading_status
                }
                
                if next_url:
                    response_data['next_url'] = next_url
                
                return JsonResponse(response_data)
            except Exception as e:
                return JsonResponse({'success': False, 'error': str(e)}, status=400)
        
        # Regular form submission (shouldn't happen with auto-save)
        return super().post(request, *args, **kwargs)


class SaveGradingNotesView(LoginRequiredMixin, View):
    """AJAX endpoint to save grading notes for a specific session."""

    def post(self, request, question_id):
        """Save grading notes for a specific session."""
        if request.headers.get('X-Requested-With') != 'XMLHttpRequest':
            return JsonResponse({'success': False, 'error': 'Invalid request'}, status=400)

        try:
            # Get session_id from POST data
            session_id = request.POST.get('session_id')
            if not session_id:
                return JsonResponse({'success': False, 'error': 'Session ID required'}, status=400)

            # Get the specific grading session
            session = ModelGradingSession.objects.get(
                id=session_id,
                grader=request.user,
                question_id=question_id
            )

            # Update grader notes
            session.grader_notes = request.POST.get('grader_notes', '')
            session.save()

            return JsonResponse({'success': True})
        except ModelGradingSession.DoesNotExist:
            return JsonResponse({'success': False, 'error': 'Session not found'}, status=404)
        except Exception as e:
            return JsonResponse({'success': False, 'error': str(e)}, status=400)


class ToggleNAForAllView(LoginRequiredMixin, View):
    """AJAX endpoint to toggle N/A for all model answers in a specific session."""

    def post(self, request, question_id):
        """Toggle N/A for all gradings in a specific session."""
        if request.headers.get('X-Requested-With') != 'XMLHttpRequest':
            return JsonResponse({'success': False, 'error': 'Invalid request'}, status=400)

        try:
            # Get session_id from POST data
            session_id = request.POST.get('session_id')
            if not session_id:
                return JsonResponse({'success': False, 'error': 'Session ID required'}, status=400)

            # Get the specific grading session
            session = ModelGradingSession.objects.get(
                id=session_id,
                grader=request.user,
                question_id=question_id
            )
            
            # Check if we're enabling or disabling N/A for all
            toggle_action = request.POST.get('toggle_na_action')
            
            if toggle_action == 'enable':
                # Set N/A for all answers in this session
                # Don't overwrite existing selections - just set to N/A (id=4)
                gradings = ModelGrading.objects.filter(session=session)
                
                # Only update answers that don't already have N/A set
                updated_gradings = gradings.exclude(achievement_correct_result_id=4)
                updated_count = updated_gradings.count()
                
                if updated_count > 0:
                    updated_gradings.update(achievement_correct_result_id=4)
                    
                    return JsonResponse({
                        'success': True,
                        'message': f'Set N/A for {updated_count} other answer(s). Correct End Result is now disabled for all.',
                        'updated_count': updated_count,
                        'action': 'enabled'
                    })
                else:
                    return JsonResponse({
                        'success': True,
                        'message': 'All answers already have N/A set.',
                        'updated_count': 0,
                        'action': 'enabled'
                    })
                    
            elif toggle_action == 'disable':
                # Don't clear existing True/False/Not Sure selections
                # Just clear N/A selections to allow editing
                gradings = ModelGrading.objects.filter(
                    session=session,
                    achievement_correct_result_id=4  # N/A state ID
                )
                
                updated_count = gradings.count()
                
                if updated_count > 0:
                    # Set to NULL only for those that were N/A
                    gradings.update(achievement_correct_result_id=None)
                    
                    return JsonResponse({
                        'success': True,
                        'message': f'Enabled Correct End Result for {updated_count} other answer(s). Please select True/False/Not Sure for each.',
                        'updated_count': updated_count,
                        'action': 'disabled'
                    })
                else:
                    return JsonResponse({
                        'success': True,
                        'message': None,
                        'updated_count': 0,
                        'action': 'disabled'
                    })
            
            return JsonResponse({'success': False, 'error': 'Invalid action'}, status=400)
            
        except ModelGradingSession.DoesNotExist:
            return JsonResponse({'success': False, 'error': 'Session not found'}, status=404)
        except Exception as e:
            return JsonResponse({'success': False, 'error': str(e)}, status=400)


class FinalizeGradingView(LoginRequiredMixin, GradingAccessMixin, View):
    """Finalize grading session and reveal model identities."""

    def post(self, request, *args, **kwargs):
        """Finalize a specific grading session."""
        # Get session_id from POST data
        session_id = request.POST.get('session_id')
        if not session_id:
            messages.error(request, "Session ID is required to finalize.")
            return redirect('questions:grading_overview', question_id=self.question.pk)

        # Get the specific session
        session = get_object_or_404(
            ModelGradingSession,
            id=session_id,
            grader=request.user,
            question=self.question
        )

        if session.is_finalized:
            messages.warning(request, "This grading session has already been finalized.")
            return redirect('questions:grading_overview', question_id=self.question.pk)

        # Check if all gradings are complete
        gradings = ModelGrading.objects.filter(session=session)
        total_count = gradings.count()
        complete_count = gradings.filter(grading_status__in=['completed', 'given_up']).count()

        if complete_count < total_count:
            messages.error(
                request,
                f"Cannot finalize: {total_count - complete_count} answer(s) still need to be graded."
            )
            return redirect('questions:grading_overview', question_id=self.question.pk)

        # Finalize the session
        with transaction.atomic():
            session.session_status = 'finalized'
            session.finalized_at = timezone.now()
            session.save()

            # Mark all gradings as finalized
            gradings.update(finalized_at=timezone.now())

        messages.success(request, "Grading session finalized! Model identities are now revealed for this session.")
        return redirect('questions:grading_overview', question_id=self.question.pk)


class GradingFocusView(GradingDetailView):
    """
    Focus mode for grading - minimal UI for distraction-free grading.

    Inherits from GradingDetailView to reuse all the grading logic,
    but uses a different template with minimal chrome.
    """
    template_name = 'questions/grading_focus.html'
    
    def get_context_data(self, **kwargs):
        """Override to generate focus-mode URLs for tabs."""
        context = super().get_context_data(**kwargs)
        
        # Update tab URLs to use focus mode
        if 'tabs' in context:
            for tab in context['tabs']:
                # Replace grading_detail URL with grading_focus URL
                if 'grading/' in tab['url']:
                    # Convert to focus mode URL
                    base_url = reverse('questions:grading_focus', kwargs={
                        'question_id': self.question.pk,
                        'alias': tab['alias']
                    })
                    # Preserve inspect_session parameter if present
                    if self.admin_inspection_mode and self.request.GET.get('inspect_session'):
                        base_url += f"?inspect_session={self.request.GET.get('inspect_session')}"
                    tab['url'] = base_url
        
        return context

    def _get_next_incomplete_url(self):
        """Get next incomplete URL, converting to focus mode if it's a detail view."""
        next_url = super()._get_next_incomplete_url()

        # If it's a grading detail URL (has alias), append /focus/
        # Overview URLs (ending with /grading or /grading/) are left as-is to exit focus mode
        if '/grading/' in next_url and not (next_url.endswith('/grading/') or next_url.endswith('/grading')):
            next_url = next_url.rstrip('/') + '/focus/'

        return next_url


class AdminGradingInspectionView(LoginRequiredMixin, TemplateView):
    """
    Admin view for inspecting all grading sessions across all questions.

    Provides administrators with a comprehensive overview of grading activity,
    including active and finalized sessions, grader progress, and statistics.
    """
    template_name = 'questions/admin_grading_inspection.html'

    def dispatch(self, request, *args, **kwargs):
        """Check if user is an admin."""
        if not (hasattr(request.user, 'participant') and request.user.participant.is_admin):
            messages.error(request, "You don't have permission to access grading inspection.")
            return redirect('home')
        return super().dispatch(request, *args, **kwargs)

    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)

        # Get all grading sessions with related data
        sessions = ModelGradingSession.objects.all().select_related(
            'grader', 'question__author'
        ).prefetch_related(
            'aliases__model_answer__model__company',
            'aliases__model_answer__model__tier',
            'gradings'
        )

        # Organize sessions by status
        active_sessions = []
        finalized_sessions = []

        for session in sessions:
            # Get grading statistics for this session
            gradings = session.gradings.all()
            total_answers = session.aliases.count()

            # Count grading statuses
            completed = gradings.filter(grading_status='completed').count()
            in_progress = gradings.filter(grading_status='in_progress').count()
            given_up = gradings.filter(grading_status='given_up').count()
            not_started = total_answers - (completed + in_progress + given_up)

            session_data = {
                'session': session,
                'total_answers': total_answers,
                'completed': completed,
                'in_progress': in_progress,
                'given_up': given_up,
                'not_started': not_started,
                'progress_percentage': round((completed + given_up) / total_answers * 100) if total_answers > 0 else 0
            }

            if session.is_finalized:
                finalized_sessions.append(session_data)
            else:
                active_sessions.append(session_data)

        # Get last activity for active sessions
        for session_data in active_sessions:
            # Get the most recent grading update for this session
            last_grading = ModelGrading.objects.filter(
                session=session_data['session']
            ).order_by('-id').first()
            
            if last_grading and hasattr(last_grading, 'updated_at'):
                session_data['last_activity'] = last_grading.updated_at
            else:
                # Fall back to session creation time if no gradings yet
                session_data['last_activity'] = session_data['session'].created_at
        
        # Sort sessions: active by last activity desc, finalized by finalized_at desc
        active_sessions.sort(key=lambda x: x.get('last_activity', x['session'].created_at), reverse=True)
        finalized_sessions.sort(key=lambda x: x['session'].finalized_at or x['session'].created_at, reverse=True)

        context['active_sessions'] = active_sessions
        context['finalized_sessions'] = finalized_sessions
        context['total_sessions'] = len(sessions)
        context['active_count'] = len(active_sessions)
        context['finalized_count'] = len(finalized_sessions)

        # Get questions with released answers but no grading sessions
        questions_with_released = Question.objects.filter(
            modelanswer__released_for_grading=True
        ).distinct()

        questions_without_sessions = []
        for question in questions_with_released:
            if not ModelGradingSession.objects.filter(question=question).exists():
                released_count = ModelAnswer.objects.filter(
                    question=question,
                    released_for_grading=True
                ).count()
                questions_without_sessions.append({
                    'question': question,
                    'released_count': released_count
                })

        context['questions_without_sessions'] = questions_without_sessions

        return context


class AdminInspectSessionView(LoginRequiredMixin, DetailView):
    """
    Admin read-only view for inspecting a specific grading session.

    Allows administrators to view the details of any grading session,
    including the grades given and the model identities (if finalized).
    """
    template_name = 'questions/admin_inspect_session.html'
    model = ModelGradingSession
    context_object_name = 'session'

    def dispatch(self, request, *args, **kwargs):
        """Check if user is an admin."""
        if not (hasattr(request.user, 'participant') and request.user.participant.is_admin):
            messages.error(request, "You don't have permission to inspect grading sessions.")
            return redirect('home')
        return super().dispatch(request, *args, **kwargs)

    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        session = self.object

        # Get all aliases with their grading status
        aliases = ModelGradingAlias.objects.filter(
            session=session
        ).select_related('model_answer__model__tier', 'model_answer__model__company')

        # Get all gradings for this session
        gradings = ModelGrading.objects.filter(
            session=session
        ).select_related('model_answer')

        # Create a mapping of model_answer_id to grading
        grading_map = {g.model_answer_id: g for g in gradings}

        # Helper function to convert GradingState to display value
        def get_state_value(state_obj):
            if state_obj is None:
                return None
            if state_obj.state_code == 'true':
                return True
            elif state_obj.state_code == 'false':
                return False
            elif state_obj.state_code == 'not_sure':
                return None  # Will show as ?
            elif state_obj.state_code == 'not_applicable':
                return 'N/A'
            return None

        # Organize aliases by tier group with grading details
        tier1_aliases = []
        tier23_aliases = []
        tier4plus_aliases = []

        for alias in aliases:
            grading = grading_map.get(alias.model_answer_id)

            alias_data = {
                'alias': alias.alias,
                'alias_order': alias.alias_order,
                'model_answer': alias.model_answer,
                'grading_status': grading.grading_status if grading else 'not_started',
                'is_complete': grading.is_complete if grading else False,
                'progress_grade': grading.get_progress_grade_display() if grading and grading.progress_grade is not None else None,
                'progress_grade_raw': grading.progress_grade if grading else None,
                'comments': grading.comments if grading else '',
                'flag_for_organizers': grading.flag_for_organizers if grading else False,
                # Add all grading fields for display
                'error_incorrect_logic': get_state_value(grading.error_incorrect_logic) if grading else None,
                'error_hallucinated': get_state_value(grading.error_hallucinated) if grading else None,
                'error_calculation': get_state_value(grading.error_calculation) if grading else None,
                'error_conceptual': get_state_value(grading.error_conceptual) if grading else None,
                'achievement_understanding': get_state_value(grading.achievement_understanding) if grading else None,
                'achievement_correct_result': get_state_value(grading.achievement_correct_result) if grading else None,
                'achievement_insight': get_state_value(grading.achievement_insight) if grading else None,
                'achievement_usefulness': get_state_value(grading.achievement_usefulness) if grading else None,
            }

            # Always include model identity for admins (not just when finalized)
            alias_data['model_name'] = alias.model_answer.model.model_name
            alias_data['company_name'] = alias.model_answer.model.company.company_name

            if alias.tier_group == 1:
                tier1_aliases.append(alias_data)
            elif alias.tier_group == 2:
                tier23_aliases.append(alias_data)
            else:  # tier_group == 3 (Tier 4+)
                tier4plus_aliases.append(alias_data)

        # Sort by alias order for consistent display
        tier1_aliases.sort(key=lambda x: x['alias_order'])
        tier23_aliases.sort(key=lambda x: x['alias_order'])
        tier4plus_aliases.sort(key=lambda x: x['alias_order'])

        context['tier1_aliases'] = tier1_aliases
        context['tier23_aliases'] = tier23_aliases
        context['tier4plus_aliases'] = tier4plus_aliases
        context['question'] = session.question

        # Calculate completion statistics
        total_count = len(aliases)
        completed_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'completed')
        in_progress_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'in_progress')
        given_up_count = sum(1 for a in aliases if grading_map.get(a.model_answer_id, None) and grading_map[a.model_answer_id].grading_status == 'given_up')

        context['total_count'] = total_count
        context['completed_count'] = completed_count
        context['in_progress_count'] = in_progress_count
        context['given_up_count'] = given_up_count
        context['not_started_count'] = total_count - completed_count - in_progress_count - given_up_count

        return context


class DeleteGradingSessionView(LoginRequiredMixin, View):
    """
    Admin view to delete a grading session.

    Allows administrators to delete grading sessions, which will cascade delete
    all associated aliases and gradings.
    """

    def post(self, request, pk):
        """Delete the grading session."""
        # Check if user is an admin
        if not (hasattr(request.user, 'participant') and request.user.participant.is_admin):
            messages.error(request, "You don't have permission to delete grading sessions.")
            return redirect('questions:grading_inspection')

        # Get the session
        session = get_object_or_404(ModelGradingSession, pk=pk)

        # Store info for message
        grader_email = session.grader.email
        question_title = session.question.title

        # Delete the session (cascade will delete aliases and gradings)
        session.delete()

        messages.success(
            request,
            f"Grading session for '{question_title}' by {grader_email} has been deleted."
        )

        return redirect('questions:grading_inspection')
