"""
Views for question management and review workflow.

Provides interfaces for:
- Question CRUD operations with role-based access control
- Review submission and response workflow
- Subquestion management (add, edit, delete, reorder)
- Model grading interface for question authors
- Version history and diff views
"""

from django.views.generic import ListView, DetailView, CreateView, UpdateView, DeleteView, TemplateView
from django.db.models import Q
from django.urls import reverse_lazy
from django.contrib import messages
from django.contrib.auth.decorators import login_required
from django.db import IntegrityError
from django.http import JsonResponse
from django.views.decorators.http import require_POST
from django.shortcuts import redirect, get_object_or_404, render
from django.core.exceptions import PermissionDenied
from django.views import View
import json
from .models import Question, QuestionState, Subquestion, EvaluationType, QuestionReview, ReviewReply, QuestionVersion
from .forms import QuestionForm, QuestionReviewForm
from accounts.permissions import ContributorRequiredMixin, OwnerOrAdminMixin
from django.contrib.auth.mixins import LoginRequiredMixin


class QuestionListView(LoginRequiredMixin, ListView):
    """
    Display a list of all questions with filtering and search functionality.
    
    Features:
    - Bootstrap-styled table display
    - Filter by title, author, and status
    - Show difficulty ratings as badges
    - Professional LaTeX-ready layout
    """
    model = Question
    template_name = 'questions/list.html'
    context_object_name = 'questions'
    paginate_by = 15  # Slightly fewer per page since questions have more metadata
    
    def get_queryset(self):
        """Filter questions based on search query, filters, and user role."""
        from django.db.models import Q
        queryset = Question.objects.select_related('author', 'status').prefetch_related('reviews').order_by('-last_modified')
        
        # Role-based filtering: contributors see their own questions + assigned reviews + published questions
        if hasattr(self.request.user, 'participant'):
            participant = self.request.user.participant
            if not participant.is_reviewer:  # Contributors (not reviewers/admins)
                from review_invitations.models import ReviewAssignment

                # Get questions assigned for review
                assigned_question_ids = ReviewAssignment.objects.filter(
                    user=self.request.user,
                    is_active=True
                ).values_list('question_id', flat=True)

                # Show own questions OR assigned questions OR published questions
                queryset = queryset.filter(
                    Q(author=participant) | Q(id__in=assigned_question_ids) | Q(published__isnull=False)
                )
        
        # Search functionality
        search_query = self.request.GET.get('search', '').strip()
        if search_query:
            queryset = queryset.filter(
                Q(title__icontains=search_query) |
                Q(author__name__icontains=search_query) |
                Q(tags__icontains=search_query)
            )
        
        # Status filter
        status_filter = self.request.GET.get('status', '').strip()
        if status_filter:
            queryset = queryset.filter(status__status=status_filter)
        
        return queryset
    
    def get_context_data(self, **kwargs):
        """Add search parameters and filter options to context."""
        context = super().get_context_data(**kwargs)
        context['search_query'] = self.request.GET.get('search', '')
        context['status_filter'] = self.request.GET.get('status', '')
        
        # Get available status options for filter dropdown
        context['available_statuses'] = QuestionState.objects.all().order_by('status')
        
        # Add total count for search results (before pagination)
        if context['search_query'] or context['status_filter']:
            # Get the full queryset before pagination to count total results
            full_queryset = self.get_queryset()
            context['total_question_count'] = full_queryset.count()
        
        # Check for review assignments for the current user
        from review_invitations.models import ReviewAssignment
        if self.request.user.is_authenticated:
            assigned_question_ids = ReviewAssignment.objects.filter(
                user=self.request.user,
                is_active=True
            ).values_list('question_id', flat=True)
            context['assigned_questions'] = set(assigned_question_ids)
        
        # Add released answer counts for each question to enable Grade button
        from model_evaluation.models import ModelAnswer, ModelGradingSession
        from django.db.models import Count, Q as DQ
        
        # Get all question IDs from the current page
        question_ids = [q.id for q in context['questions']]
        
        # Get counts of released answers for these questions
        released_counts = ModelAnswer.objects.filter(
            question_id__in=question_ids,
            released_for_grading=True
        ).values('question_id').annotate(count=Count('id'))
        
        # Create a mapping of question_id to released count
        released_map = {item['question_id']: item['count'] for item in released_counts}
        
        # Check for finalized grading sessions for these questions
        finalized_sessions = ModelGradingSession.objects.filter(
            question_id__in=question_ids,
            session_status='finalized'
        ).values('question_id').annotate(count=Count('id'))
        
        # Create a mapping of question_id to whether it has finalized grading
        finalized_map = {item['question_id']: item['count'] > 0 for item in finalized_sessions}
        
        # Add released_answer_count to each question and can_grade flag
        user = self.request.user

        # Get all active review assignments for current user (for efficient querying)
        assigned_question_ids = set()
        if user.is_authenticated:
            from review_invitations.models import ReviewAssignment
            assigned_question_ids = set(
                ReviewAssignment.objects.filter(
                    user=user,
                    is_active=True
                ).values_list('question_id', flat=True)
            )

        for question in context['questions']:
            question.released_answer_count = released_map.get(question.id, 0)
            question.has_finalized_grading = finalized_map.get(question.id, False)

            # User can grade if they are the author, assigned reviewer, or admin
            is_author = hasattr(user, 'participant') and question.author == user.participant
            is_admin = hasattr(user, 'participant') and user.participant.is_admin
            is_assigned_reviewer = question.id in assigned_question_ids

            question.can_grade = (
                question.released_answer_count > 0 and
                (is_author or is_assigned_reviewer or is_admin)
            )
        
        return context


class QuestionDetailView(LoginRequiredMixin, DetailView):
    """
    Display detailed view of a single question with rendered LaTeX content.
    
    Features:
    - Full question text with LaTeX rendering
    - Solution display (if available)
    - Metadata display (difficulty, tags, timestamps)
    - Related subquestions and grading schemes
    - Clean, readable layout optimized for mathematical content
    """
    model = Question
    template_name = 'questions/detail.html'
    context_object_name = 'question'
    
    def get_queryset(self):
        """Optimize query with related objects."""
        return Question.objects.select_related('author', 'status').prefetch_related(
            'subquestion_set'
        )
    
    def get_object(self, queryset=None):
        """Override to add permission checking."""
        obj = super().get_object(queryset)

        # Published questions are visible to all logged-in users
        if obj.published:
            return obj

        # Check if user has permission to view this question
        if hasattr(self.request.user, 'participant'):
            participant = self.request.user.participant

            # Admins and reviewers can see all questions
            if participant.is_reviewer or participant.is_admin:
                return obj

            # Contributors can only see their own questions or assigned reviews
            from review_invitations.models import ReviewAssignment
            is_own_question = obj.author == participant
            is_assigned_review = ReviewAssignment.objects.filter(
                user=self.request.user,
                question=obj,
                is_active=True
            ).exists()

            if is_own_question or is_assigned_review:
                return obj

        # If we get here, user doesn't have permission
        raise PermissionDenied("You don't have permission to view this question.")
    
    def get_context_data(self, **kwargs):
        """Add related objects to context."""
        context = super().get_context_data(**kwargs)
        
        # Get subquestions ordered by subquestion_order
        context['subquestions'] = self.object.subquestion_set.all().order_by('subquestion_order')
        
        # Check if there are released model answers for grading
        from model_evaluation.models import ModelAnswer
        context['has_released_answers'] = ModelAnswer.objects.filter(
            question=self.object,
            released_for_grading=True
        ).exists()
        
        # Check if user can grade (author, assigned reviewer, or admin)
        user = self.request.user
        is_author = hasattr(user, 'participant') and self.object.author == user.participant
        is_admin = hasattr(user, 'participant') and user.participant.is_admin

        # Check if user has accepted a review invitation for this question
        is_assigned_reviewer = False
        if user.is_authenticated:
            from review_invitations.models import ReviewAssignment
            is_assigned_reviewer = ReviewAssignment.objects.filter(
                user=user,
                question=self.object,
                is_active=True
            ).exists()

        context['can_grade'] = (
            user.is_authenticated and
            context['has_released_answers'] and
            (is_author or is_assigned_reviewer or is_admin)
        )
        
        # Get latest AI test result for display
        try:
            from ai_testing.services import OpenAITestService
            service = OpenAITestService()
            latest_test = service.get_latest_successful_test(self.object)
            context['latest_ai_test'] = latest_test
        except Exception:
            # If AI testing app is not available, just skip
            context['latest_ai_test'] = None
        
        # Check if current user is the author
        if hasattr(self.request.user, 'participant'):
            context['is_author'] = self.object.author == self.request.user.participant
            context['is_reviewer'] = self.request.user.participant.is_reviewer
            context['is_admin'] = self.request.user.participant.is_admin
        else:
            context['is_author'] = False
            context['is_reviewer'] = False
            context['is_admin'] = False

        # Check for grading sessions (for admin inspection)
        from model_evaluation.models import ModelGradingSession
        context['has_grading_sessions'] = False
        context['active_grading_sessions'] = []
        context['finalized_grading_sessions'] = []

        if context['is_admin']:
            # Get all grading sessions for this question
            # Use select_related to avoid N+1 queries when accessing grader and participant
            sessions = ModelGradingSession.objects.filter(
                question_id=self.object.id
            ).select_related('grader', 'grader__participant')

            for session in sessions:
                # Get grader email or name safely
                grader_display = 'Unknown Grader'
                try:
                    # session.grader IS the User model directly
                    if session.grader:
                        grader_display = session.grader.email
                        # If the user has a participant with a name, include it
                        if hasattr(session.grader, 'participant') and session.grader.participant:
                            if session.grader.participant.name:
                                grader_display = f"{session.grader.participant.name} ({session.grader.email})"
                except:
                    pass

                session_info = {
                    'session': session,
                    'grader_email': grader_display
                }

                if session.finalized_at:
                    context['finalized_grading_sessions'].append(session_info)
                else:
                    context['active_grading_sessions'].append(session_info)

            context['has_grading_sessions'] = len(context['active_grading_sessions']) > 0 or len(context['finalized_grading_sessions']) > 0
        
        # Check if user can submit for review
        # Authors can submit their own questions, admins can submit any question
        context['can_submit_for_review'] = (
            (context['is_author'] or context['is_admin']) and 
            self.object.status.status == 'draft'
        )
        
        # Get count of assigned reviewers for this question
        from review_invitations.models import ReviewAssignment
        context['reviewer_count'] = ReviewAssignment.objects.filter(
            question=self.object,
            is_active=True
        ).count()
        
        # Check if user can review this question
        # Reviewers OR contributors with review assignment can review
        from review_invitations.models import ReviewAssignment, ReviewInvitation
        is_assigned_reviewer = False
        if self.request.user.is_authenticated:
            is_assigned_reviewer = ReviewAssignment.objects.filter(
                user=self.request.user,
                question=self.object,
                is_active=True
            ).exists()
        
        context['can_review'] = (
            (context['is_reviewer'] or is_assigned_reviewer) and 
            not context['is_author'] and  # Can't review own questions
            self.object.status.status == 'under_review'
        )
        
        # Check if user has an accepted invitation (to show decline button)
        context['has_accepted_invitation'] = False
        if self.request.user.is_authenticated:
            context['has_accepted_invitation'] = ReviewInvitation.objects.filter(
                assignments__user=self.request.user,
                assignments__question=self.object,
                assignments__is_active=True,
                status='accepted'
            ).exists()
        
        # Count how many times this user has reviewed this question
        if hasattr(self.request.user, 'participant') and context['can_review']:
            review_count = QuestionReview.objects.filter(
                question=self.object,
                reviewer=self.request.user.participant
            ).count()
            context['user_review_count'] = review_count
        else:
            context['user_review_count'] = 0
        
        # Get reviews that this user can see, separated into open and closed
        open_reviews = []
        closed_reviews = []
        
        if hasattr(self.request.user, 'participant'):
            participant = self.request.user.participant
            all_reviews = self.object.reviews.all().prefetch_related('replies').order_by('-time')
            
            for review in all_reviews:
                # Check if user can view this review
                if (review.question.author == participant or  # Author can see reviews of their questions
                    review.reviewer == participant or        # Reviewer can see their own reviews  
                    participant.is_admin):                   # Admins can see all reviews
                    
                    # Add action flags for template
                    has_replies = review.replies.exists()
                    is_reviewer = review.reviewer == participant
                    is_revise_decision = review.decision == 'revise'
                    question_under_review = self.object.status.status == 'under_review'
                    
                    # Can perform reviewer actions (accept/close/add review)
                    review.can_accept = (is_reviewer and is_revise_decision and 
                                       question_under_review and has_replies and 
                                       not review.is_concluded)
                    
                    review.can_close = review.can_accept  # Same conditions
                    review.can_add_review = (is_reviewer and has_replies and 
                                           not review.is_concluded)
                    
                    # Can reply to this specific review (author only, and only if no reply from author yet)
                    is_author = review.question.author == participant
                    author_has_replied = False
                    if has_replies:
                        author_has_replied = review.replies.filter(author=review.question.author).exists()
                    
                    review.can_reply = (is_author and not author_has_replied)
                    
                    # New categorization logic matching model methods:
                    # Open: No replies OR (has replies + not concluded + decision not in ['accept', 'reject'])
                    # Closed: Has replies AND (concluded OR decision in ['accept', 'reject'])
                    if not has_replies:
                        open_reviews.append(review)  # No replies = definitely open
                    elif not review.is_concluded and review.decision not in ['accept', 'reject']:
                        open_reviews.append(review)  # Actionable review = open
                    else:
                        closed_reviews.append(review)  # Everything else = closed
        
        context['open_reviews'] = open_reviews
        context['closed_reviews'] = closed_reviews
        context['reviews'] = open_reviews + closed_reviews  # Keep for backward compatibility
        
        return context




class QuestionCreateView(ContributorRequiredMixin, CreateView):
    """
    View for creating new questions with django-martor editor.
    
    Features:
    - Professional form layout with martor integration
    - LaTeX + Markdown editing with live preview
    - Comprehensive validation and error handling
    - Draft saving capabilities
    """
    model = Question
    form_class = QuestionForm
    template_name = 'questions/form.html'
    success_url = reverse_lazy('questions:list')
    
    def get_form_kwargs(self):
        """Pass the current user to the form."""
        kwargs = super().get_form_kwargs()
        kwargs['user'] = self.request.user
        return kwargs
    
    def form_valid(self, form):
        """Handle successful form submission."""
        try:
            # Set the author to current user's participant
            if hasattr(self.request.user, 'participant'):
                form.instance.author = self.request.user.participant
            
            # Check if saving as draft
            if 'save_draft' in self.request.POST:
                # Set status to draft if saving as draft
                draft_status = QuestionState.objects.filter(status='draft').first()
                if draft_status:
                    form.instance.status = draft_status
            
            # Save the question first
            response = super().form_valid(form)

            # Process subquestions
            self.process_subquestions()

            # Create initial version (v1) for the new question
            created_by = getattr(self.request.user, 'participant', None)
            QuestionVersion.create_version(self.object, created_by=created_by)

            # Success message
            action = "saved as draft" if 'save_draft' in self.request.POST else "created"
            subquestion_count = self.object.subquestion_set.count()
            subq_msg = f" with {subquestion_count} subquestions" if subquestion_count > 0 else ""

            messages.success(
                self.request,
                f'✅ Question "{self.object.title}" has been {action} successfully{subq_msg}!'
            )

            # Redirect to detail view if not draft
            if 'save_draft' not in self.request.POST:
                self.success_url = reverse_lazy('questions:detail', kwargs={'pk': self.object.pk})

            return response

        except IntegrityError as e:
            # Handle database constraint violations
            messages.error(
                self.request,
                '❌ Unable to save question. Please check for duplicate titles or other constraint violations.'
            )
            return self.form_invalid(form)
    
    def process_subquestions(self):
        """Process subquestion data from the form."""
        # Find all subquestion data in POST
        subquestion_data = {}
        
        for key, value in self.request.POST.items():
            if key.startswith('subquestion_') and value.strip():
                # Parse field name: subquestion_0_text, subquestion_1_answer, etc.
                parts = key.split('_')
                if len(parts) >= 3:
                    index = parts[1]
                    field = '_'.join(parts[2:])  # Handle fields like evaluation_type
                    
                    if index not in subquestion_data:
                        subquestion_data[index] = {}
                    subquestion_data[index][field] = value
        
        # Create subquestions
        for i, (index, data) in enumerate(sorted(subquestion_data.items())):
            # Only create if we have required fields
            if data.get('text') and data.get('answer'):
                try:
                    # Get evaluation type
                    eval_type = EvaluationType.objects.get(pk=data.get('evaluation_type', 1))
                    
                    Subquestion.objects.create(
                        question=self.object,
                        subquestion_order=chr(ord('a') + i),  # a, b, c, etc.
                        text=data['text'],
                        answer=data['answer'],
                        rationale=data.get('rationale', ''),
                        points=int(data.get('points', 1)) if data.get('points') else 1,
                        evaluation_type=eval_type
                    )
                except (ValueError, EvaluationType.DoesNotExist) as e:
                    # Skip invalid subquestions but don't fail the whole process
                    continue
    
    def form_invalid(self, form):
        """Handle form validation errors."""
        messages.error(
            self.request,
            '❌ Please correct the errors below and try again.'
        )
        return super().form_invalid(form)
    
    def post(self, request, *args, **kwargs):
        """Handle POST requests - either AJAX save or regular form submission."""
        # Check if this is an AJAX request
        if request.headers.get('X-Requested-With') == 'XMLHttpRequest':
            return self.ajax_save_question(request, *args, **kwargs)
        else:
            # Regular form submission
            return super().post(request, *args, **kwargs)
    
    def ajax_save_question(self, request, *args, **kwargs):
        """
        AJAX endpoint for saving new question without page refresh.
        
        Returns JSON response with success/error status and new question ID.
        """
        self.object = None  # For CreateView
        form = self.get_form()
        
        if form.is_valid():
            try:
                # Save the question first
                response = super().form_valid(form)

                # Process subquestions
                self.process_subquestions()

                # Create initial version (v1) for the new question
                created_by = getattr(request.user, 'participant', None)
                QuestionVersion.create_version(self.object, created_by=created_by)

                # Return success response with new question ID
                subquestion_count = self.object.subquestion_set.count()
                subq_msg = f" with {subquestion_count} subquestions" if subquestion_count > 0 else ""

                return JsonResponse({
                    'success': True,
                    'message': f'✅ Question created successfully{subq_msg}!',
                    'question_id': self.object.pk,
                    'edit_url': reverse_lazy('questions:edit', kwargs={'pk': self.object.pk}),
                    'timestamp': self.object.last_modified.strftime('%Y-%m-%d %H:%M:%S') if hasattr(self.object, 'last_modified') else ''
                })

            except IntegrityError as e:
                return JsonResponse({
                    'success': False,
                    'message': '❌ Unable to create question. Please check for constraint violations.',
                    'errors': {'database': str(e)}
                })
                
        else:
            # Return form validation errors
            return JsonResponse({
                'success': False,
                'message': '❌ Please correct the errors below and try again.',
                'errors': form.errors
            })


class QuestionDeleteView(OwnerOrAdminMixin, DeleteView):
    """
    View for deleting questions with confirmation.
    
    Features:
    - Confirmation page to prevent accidental deletion
    - Shows all related data that will be deleted
    - Automatic cleanup of related data
    - Success message and redirect to questions list
    """
    model = Question
    template_name = 'questions/delete.html'
    success_url = reverse_lazy('questions:list')
    context_object_name = 'question'
    
    def delete(self, request, *args, **kwargs):
        """Handle the deletion with a success message."""
        from model_evaluation.models import (
            ModelAttempt, ModelAnswer, EvaluationQueue,
            ModelGradingSession, ModelGradingAlias, ModelGrading,
            ModelSubquestionAnswer, ExecutionTracker
        )
        from review_invitations.models import ReviewAssignment
        from ai_testing.models import TestAttempt, QuestionDailyLimit
        
        self.object = self.get_object()
        question_title = self.object.title or f"Question {self.object.id}"
        
        # Collect counts for the success message
        subquestion_count = self.object.subquestion_set.count()
        review_count = self.object.reviews.count()
        model_attempt_count = ModelAttempt.objects.filter(question_id=self.object.id).count()
        model_answer_count = ModelAnswer.objects.filter(question_id=self.object.id).count()
        review_assignment_count = ReviewAssignment.objects.filter(question=self.object).count()
        test_attempt_count = TestAttempt.objects.filter(question=self.object).count()
        
        try:
            from django.db import transaction
            
            with transaction.atomic():
                # Delete grading data first (references model_answers)
                grading_sessions = ModelGradingSession.objects.filter(question_id=self.object.id)
                for session in grading_sessions:
                    ModelGrading.objects.filter(session=session).delete()
                    ModelGradingAlias.objects.filter(session=session).delete()
                grading_sessions.delete()
                
                # Delete model evaluation data (unmanaged models - need manual deletion)
                # Delete in dependency order
                ModelSubquestionAnswer.objects.filter(
                    attempt__question_id=self.object.id
                ).delete()
                
                ExecutionTracker.objects.filter(question_id=self.object.id).delete()
                
                EvaluationQueue.objects.filter(
                    attempt__question_id=self.object.id
                ).delete()
                
                ModelAnswer.objects.filter(question_id=self.object.id).delete()
                ModelAttempt.objects.filter(question_id=self.object.id).delete()
                
                # Delete review assignments (managed model)
                ReviewAssignment.objects.filter(question=self.object).delete()
                
                # Delete AI testing data (managed models)
                QuestionDailyLimit.objects.filter(question=self.object).delete()
                TestAttempt.objects.filter(question=self.object).delete()
                
                # Delete the question itself
                # This will cascade to: subquestions, question_reviews, review_replies
                # due to foreign key CASCADE settings
                self.object.delete()
            
            # Build success message with details
            related_info = []
            if subquestion_count > 0:
                related_info.append(f"{subquestion_count} subquestion{'s' if subquestion_count != 1 else ''}")
            if review_count > 0:
                related_info.append(f"{review_count} review{'s' if review_count != 1 else ''}")
            if model_attempt_count > 0:
                related_info.append(f"{model_attempt_count} model evaluation{'s' if model_attempt_count != 1 else ''}")
            if review_assignment_count > 0:
                related_info.append(f"{review_assignment_count} review assignment{'s' if review_assignment_count != 1 else ''}")
            if test_attempt_count > 0:
                related_info.append(f"{test_attempt_count} AI test{'s' if test_attempt_count != 1 else ''}")
            
            related_msg = f" along with {', '.join(related_info)}" if related_info else ""
            
            messages.success(
                request,
                f'✅ Question "{question_title}" has been deleted successfully{related_msg}.'
            )
            
            return redirect(self.success_url)
            
        except Exception as e:
            messages.error(
                request,
                f'❌ Unable to delete question "{question_title}": {str(e)}'
            )
            return redirect('questions:detail', pk=self.object.pk)
    
    def get_context_data(self, **kwargs):
        """Add comprehensive context about related data for the confirmation page."""
        from model_evaluation.models import ModelAttempt, ModelAnswer, ModelGradingSession, ModelGradingAlias, ModelGrading
        from review_invitations.models import ReviewAssignment
        from ai_testing.models import TestAttempt
        
        context = super().get_context_data(**kwargs)
        
        # Add counts of all related objects for the confirmation page
        context['subquestion_count'] = self.object.subquestion_set.count()
        context['review_count'] = self.object.reviews.count()
        context['review_reply_count'] = ReviewReply.objects.filter(review__question=self.object).count()
        
        # Model evaluation related counts
        context['model_attempt_count'] = ModelAttempt.objects.filter(question_id=self.object.id).count()
        context['model_answer_count'] = ModelAnswer.objects.filter(question_id=self.object.id).count()
        
        # Grading related counts
        grading_sessions = ModelGradingSession.objects.filter(question_id=self.object.id)
        context['grading_session_count'] = grading_sessions.count()
        
        grading_alias_count = 0
        grading_count = 0
        for session in grading_sessions:
            grading_alias_count += ModelGradingAlias.objects.filter(session=session).count()
            grading_count += ModelGrading.objects.filter(session=session).count()
        
        context['grading_alias_count'] = grading_alias_count
        context['grading_count'] = grading_count
        
        # Review assignment count
        context['review_assignment_count'] = ReviewAssignment.objects.filter(question=self.object).count()
        
        # AI testing count
        context['test_attempt_count'] = TestAttempt.objects.filter(question=self.object).count()
        
        # Check if question is active
        context['is_active'] = self.object.status and self.object.status.status == 'active'
        
        return context


class QuestionUpdateView(OwnerOrAdminMixin, UpdateView):
    """
    View for editing existing questions with django-martor editor.
    
    Features:
    - Pre-populated form with existing question data
    - Same martor editing capabilities as create view
    - Proper handling of database triggers for timestamps
    """
    model = Question
    form_class = QuestionForm
    template_name = 'questions/form.html'
    context_object_name = 'question'
    
    def get_form_kwargs(self):
        """Pass the current user to the form."""
        kwargs = super().get_form_kwargs()
        kwargs['user'] = self.request.user
        return kwargs
    
    def get_success_url(self):
        """Redirect to question detail view after successful edit."""
        return reverse_lazy('questions:detail', kwargs={'pk': self.object.pk})
    
    def form_valid(self, form):
        """Handle successful form submission."""
        try:
            # Check if saving as draft
            if 'save_draft' in self.request.POST:
                draft_status = QuestionState.objects.filter(status='draft').first()
                if draft_status:
                    form.instance.status = draft_status
            
            # Save the question first
            response = super().form_valid(form)

            # Process subquestions (clear existing and recreate)
            self.process_subquestions()

            # Create a new version capturing this edit
            created_by = getattr(self.request.user, 'participant', None)
            QuestionVersion.create_version(self.object, created_by=created_by)

            # Success message
            action = "saved as draft" if 'save_draft' in self.request.POST else "updated"
            subquestion_count = self.object.subquestion_set.count()
            subq_msg = f" with {subquestion_count} subquestions" if subquestion_count > 0 else ""

            messages.success(
                self.request,
                f'✅ Question "{self.object.title}" has been {action} successfully{subq_msg}!'
            )

            return response

        except IntegrityError as e:
            # Handle database constraint violations
            messages.error(
                self.request,
                '❌ Unable to update question. Please check for constraint violations.'
            )
            return self.form_invalid(form)
    
    def process_subquestions(self):
        """Process subquestion data from the form (for editing) with ID-based selective update and order persistence."""
        # Parse posted subquestion fields grouped by index
        posted_by_index = {}
        for key, value in self.request.POST.items():
            if key.startswith('subquestion_'):
                parts = key.split('_')
                if len(parts) >= 3:
                    index = parts[1]
                    field = '_'.join(parts[2:])
                    posted_by_index.setdefault(index, {})[field] = value

        # Build normalized posted list in UI order (index ascending) and assign desired order letters
        posted_list = []
        for i, (index, data) in enumerate(sorted(posted_by_index.items(), key=lambda kv: int(kv[0]) if kv[0].isdigit() else kv[0])):
            text = (data.get('text') or '').strip()
            answer = (data.get('answer') or '').strip()
            if not text or not answer:
                continue
            try:
                eval_type_id = int(data.get('evaluation_type', 1)) if data.get('evaluation_type') else 1
            except (TypeError, ValueError):
                eval_type_id = 1
            try:
                points_val = int(data.get('points', 1)) if data.get('points') else 1
            except (TypeError, ValueError):
                points_val = 1
            try:
                sq_id = int(data.get('id')) if data.get('id') else None
            except (TypeError, ValueError):
                sq_id = None
            posted_list.append({
                'id': sq_id,
                'subquestion_order': chr(ord('a') + i),
                'text': text,
                'answer': answer,
                'rationale': (data.get('rationale') or '').strip(),
                'points': points_val,
                'evaluation_type_id': eval_type_id,
            })

        # Load existing subquestions by ID for selective updates
        existing_subqs = list(self.object.subquestion_set.all().order_by('subquestion_order'))
        existing_by_id = {sq.id: sq for sq in existing_subqs}

        # Track which existing IDs to keep
        kept_ids = set()

        # Apply updates/creates
        for item in posted_list:
            try:
                eval_type = EvaluationType.objects.get(pk=item['evaluation_type_id'])
            except EvaluationType.DoesNotExist:
                continue

            sq_id = item['id']
            if sq_id and sq_id in existing_by_id:
                # Update existing, preserve ID, update order
                sq = existing_by_id[sq_id]
                sq.subquestion_order = item['subquestion_order']
                sq.text = item['text']
                sq.answer = item['answer']
                sq.rationale = item['rationale']
                sq.points = item['points']
                sq.evaluation_type = eval_type
                sq.save()
                kept_ids.add(sq_id)
            else:
                # Create new
                new_sq = Subquestion.objects.create(
                    question=self.object,
                    subquestion_order=item['subquestion_order'],
                    text=item['text'],
                    answer=item['answer'],
                    rationale=item['rationale'],
                    points=item['points'],
                    evaluation_type=eval_type
                )
                kept_ids.add(new_sq.id)

        # Delete any existing subquestions not present in the posted data
        existing_ids = set(existing_by_id.keys())
        ids_to_delete = existing_ids - kept_ids
        if ids_to_delete:
            self.object.subquestion_set.filter(id__in=ids_to_delete).delete()
    
    def form_invalid(self, form):
        """Handle form validation errors."""
        messages.error(
            self.request,
            '❌ Please correct the errors below and try again.'
        )
        return super().form_invalid(form)
    
    def get_context_data(self, **kwargs):
        """Add additional context for the template."""
        context = super().get_context_data(**kwargs)
        context['form_title'] = f'Edit Question: {self.object.title}'
        context['form_subtitle'] = 'Update question content and metadata'
        
        # Add existing subquestions for JavaScript to populate
        existing_subquestions = list(
            self.object.subquestion_set.all().order_by('subquestion_order').values(
                'id', 'subquestion_order', 'text', 'answer', 'rationale', 'points', 'evaluation_type_id'
            )
        )
        context['existing_subquestions_json'] = json.dumps(existing_subquestions)

        # Add warning context when there are recorded subquestion evaluations
        try:
            from model_evaluation.models import ModelSubquestionAnswer
            subq_ids = list(self.object.subquestion_set.values_list('id', flat=True))
            subq_eval_count = 0
            if subq_ids:
                subq_eval_count = ModelSubquestionAnswer.objects.filter(subquestion_id__in=subq_ids).count()
            context['subquestion_evaluation_count'] = subq_eval_count
            context['has_subquestion_evaluations'] = subq_eval_count > 0
        except Exception:
            context['subquestion_evaluation_count'] = 0
            context['has_subquestion_evaluations'] = False
        
        # Get latest AI test result for display
        try:
            from ai_testing.services import OpenAITestService
            service = OpenAITestService()
            latest_test = service.get_latest_successful_test(self.object)
            context['latest_ai_test'] = latest_test
        except Exception:
            # If AI testing app is not available, just skip
            context['latest_ai_test'] = None
        
        return context
    
    def post(self, request, *args, **kwargs):
        """Handle POST requests - either AJAX draft save or regular form submission."""
        # Check if this is an AJAX request for draft saving
        if 'ajax-save-draft' in request.path or request.headers.get('X-Requested-With') == 'XMLHttpRequest':
            return self.ajax_save_draft(request, *args, **kwargs)
        else:
            # Regular form submission
            return super().post(request, *args, **kwargs)
    
    def ajax_save_draft(self, request, *args, **kwargs):
        """
        AJAX endpoint for saving draft without page refresh.
        
        Returns JSON response with success/error status.
        """
        self.object = self.get_object()
        form = self.get_form()
        
        if form.is_valid():
            try:
                # Save the question (keep existing status)
                form.save()

                # Process subquestions
                self.process_subquestions()

                # Create a new version capturing this edit
                created_by = getattr(request.user, 'participant', None)
                QuestionVersion.create_version(self.object, created_by=created_by)

                # Return success response
                subquestion_count = self.object.subquestion_set.count()
                subq_msg = f" with {subquestion_count} subquestions" if subquestion_count > 0 else ""

                return JsonResponse({
                    'success': True,
                    'message': f'✅ Question saved successfully{subq_msg}!',
                    'timestamp': self.object.last_modified.strftime('%Y-%m-%d %H:%M:%S') if hasattr(self.object, 'last_modified') else ''
                })

            except IntegrityError as e:
                return JsonResponse({
                    'success': False,
                    'message': '❌ Unable to save question. Please check for constraint violations.',
                    'errors': {'database': str(e)}
                })

        else:
            # Return form validation errors
            return JsonResponse({
                'success': False,
                'message': '❌ Please correct the errors below and try again.',
                'errors': form.errors
            })

class SubmitForReviewView(LoginRequiredMixin, View):
    """
    Submit a question for review by changing its status from draft to under_review.
    
    Authors can submit their own draft questions, and admins can submit any draft question.
    """
    
    def post(self, request, pk):
        """Handle the submission."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check permissions
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile to submit questions.')
            return redirect('questions:detail', pk=pk)
        
        # Authors can submit their own questions, admins can submit any question
        if question.author != request.user.participant and not request.user.participant.is_admin:
            messages.error(request, '❌ You can only submit your own questions for review.')
            return redirect('questions:detail', pk=pk)
        
        if question.status.status != 'draft':
            messages.error(request, '❌ Only draft questions can be submitted for review.')
            return redirect('questions:detail', pk=pk)
        
        # Change status to under_review
        try:
            under_review_status = QuestionState.objects.get(status='under_review')
            question.status = under_review_status
            question.save()
            
            messages.success(
                request, 
                f'✅ Question "{question.title}" has been submitted for review! '
                'You will be notified when a reviewer provides feedback.'
            )
            
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Unable to find the review status. Please contact an administrator.')
            
        return redirect('questions:detail', pk=pk)


class SubmitReviewView(LoginRequiredMixin, View):
    """
    Submit a review for a question that is under review.
    
    Only reviewers can submit reviews, and they cannot review their own questions.
    """
    
    def get(self, request, pk):
        """Show the review form."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check permissions
        if not self.can_review_question(request.user, question):
            messages.error(request, '❌ You do not have permission to review this question.')
            return redirect('questions:detail', pk=pk)
        
        # Check if this is responding to a specific review
        responding_to_id = request.GET.get('responding_to')
        responding_to_review = None
        if responding_to_id:
            try:
                responding_to_review = QuestionReview.objects.get(
                    id=responding_to_id, 
                    question=question,
                    reviewer=request.user.participant  # Must be responding to own review
                )
            except QuestionReview.DoesNotExist:
                messages.warning(request, '⚠️ Original review not found. Creating new review instead.')
        
        # Check if user has an accepted invitation (for decline button)
        from review_invitations.models import ReviewInvitation
        has_accepted_invitation = ReviewInvitation.objects.filter(
            assignments__user=request.user,
            assignments__question=question,
            assignments__is_active=True,
            status='accepted'
        ).exists()
        
        # Create the review form
        form = QuestionReviewForm(show_decline=has_accepted_invitation)
        
        context = {
            'question': question,
            'form': form,
            'subquestions': question.subquestion_set.all().order_by('subquestion_order'),
            'responding_to_review': responding_to_review,
            'has_accepted_invitation': has_accepted_invitation,
        }
        
        return render(request, 'questions/review_form.html', context)
    
    def post(self, request, pk):
        """Handle review submission."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check permissions
        if not self.can_review_question(request.user, question):
            messages.error(request, '❌ You do not have permission to review this question.')
            return redirect('questions:detail', pk=pk)
        
        # No longer checking if user already reviewed - multiple reviews allowed
        
        # Process the form
        form = QuestionReviewForm(request.POST)
        
        if form.is_valid():
            try:
                # Check if this is responding to a specific review
                responding_to_id = request.GET.get('responding_to')
                responding_to_review = None
                if responding_to_id:
                    try:
                        responding_to_review = QuestionReview.objects.get(
                            id=responding_to_id, 
                            question=question,
                            reviewer=request.user.participant
                        )
                    except QuestionReview.DoesNotExist:
                        pass
                
                # Create the new review
                review = form.save(commit=False)
                review.question = question
                review.reviewer = request.user.participant
                review.save()
                
                # If this is responding to a specific review, conclude that review
                if responding_to_review:
                    responding_to_review.is_concluded = True
                    responding_to_review.save()
                
                # Handle automatic state transitions based on review decision
                if review.decision:
                    self.handle_state_transition(question, review.decision)
                
                # Success message
                decision_text = review.get_decision_display() if review.decision else "feedback provided"
                state_msg = ""
                if review.decision == 'accept':
                    state_msg = " The question has been recommended for acceptance and awaits admin approval."
                elif review.decision == 'reject':
                    state_msg = " The question has been returned to draft status."
                elif review.decision == 'revise':
                    state_msg = " The question remains under review for further discussion."
                
                # Customize message based on whether this concludes a previous review
                if responding_to_review:
                    messages.success(
                        request,
                        f'✅ Your new review has been submitted and your previous review conversation has been concluded! '
                        f'Decision: {decision_text}.{state_msg} The author will be notified of your feedback.'
                    )
                else:
                    messages.success(
                        request,
                        f'✅ Your review has been submitted successfully! '
                        f'Decision: {decision_text}.{state_msg} The author will be notified of your feedback.'
                    )
                
                # Check if user needs to set a password (external reviewer)
                if not request.user.has_usable_password():
                    messages.info(
                        request,
                        'You can now set a password to secure your account and contribute questions to the benchmark.'
                    )
                    return redirect('review_invitations:set_password')
                
                return redirect('questions:detail', pk=pk)
                
            except Exception as e:
                messages.error(
                    request,
                    '❌ Unable to submit review. Please try again.'
                )
        else:
            # Form has validation errors
            messages.error(
                request,
                '❌ Please correct the errors below and try again.'
            )
        
        # Re-render form with errors
        context = {
            'question': question,
            'form': form,
            'subquestions': question.subquestion_set.all().order_by('subquestion_order'),
        }
        
        return render(request, 'questions/review_form.html', context)
    
    def can_review_question(self, user, question):
        """Check if user can review this question."""
        if not user.is_authenticated:
            return False
        
        if not hasattr(user, 'participant'):
            return False
        
        participant = user.participant
        
        # Check if user is a reviewer OR has been assigned to review this question
        from review_invitations.models import ReviewAssignment
        is_reviewer = participant.is_reviewer
        is_assigned = ReviewAssignment.objects.filter(
            user=user,
            question=question,
            is_active=True
        ).exists()
        
        if not (is_reviewer or is_assigned):
            return False
        
        # Cannot review own questions
        if question.author == participant:
            return False
        
        # Question must be under review
        if question.status.status != 'under_review':
            return False
        
        return True
    
    def handle_state_transition(self, question, decision):
        """Handle automatic state transitions based on review decision."""
        try:
            if decision == 'accept':
                # Accept (recommend) decision → question stays under review
                # Only admins can actually approve questions
                # No status change needed - question remains under_review for admin decision
                pass
            elif decision == 'reject':
                # Reject decision → move back to draft (author can revise privately)
                draft_status = QuestionState.objects.get(status='draft')
                question.status = draft_status
                question.save()
            elif decision == 'revise':
                # Revise decision → stay under review (review process continues)
                # No status change needed - question remains under_review
                pass
        except QuestionState.DoesNotExist:
            # If status doesn't exist, just log and continue
            pass


class RevertToDraftView(LoginRequiredMixin, View):
    """
    Allow authors to revert their questions from under_review back to draft status.
    
    This allows authors to address review feedback by making changes.
    """
    
    def post(self, request, pk):
        """Handle the reversion."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check permissions
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile to revert questions.')
            return redirect('questions:detail', pk=pk)
        
        if question.author != request.user.participant:
            messages.error(request, '❌ You can only revert your own questions.')
            return redirect('questions:detail', pk=pk)
        
        if question.status.status != 'under_review':
            messages.error(request, '❌ Only questions under review can be reverted to draft.')
            return redirect('questions:detail', pk=pk)
        
        # Change status to draft
        try:
            draft_status = QuestionState.objects.get(status='draft')
            question.status = draft_status
            question.save()
            
            messages.success(
                request, 
                f'✅ Question "{question.title}" has been reverted to draft status. '
                'You can now make changes and resubmit for review when ready.'
            )
            
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Unable to find the draft status. Please contact an administrator.')
            
        return redirect('questions:detail', pk=pk)


class ReviewGuidelinesView(LoginRequiredMixin, TemplateView):
    """
    Display review guidelines for reviewers.
    
    Provides comprehensive guidance on what makes a good benchmark question
    and how to provide constructive feedback.
    """
    template_name = 'questions/review_guidelines.html'


class SubmitReplyView(LoginRequiredMixin, View):
    """
    Submit a reply to a review.
    
    Only authors can reply to reviews of their questions.
    """
    
    def post(self, request, review_id):
        """Handle reply submission via AJAX."""
        review = get_object_or_404(QuestionReview, pk=review_id)
        
        # Check permissions - only author can reply
        if not hasattr(request.user, 'participant'):
            return JsonResponse({
                'success': False,
                'error': 'You must have a participant profile to reply to reviews.'
            }, status=403)
        
        if review.question.author != request.user.participant:
            return JsonResponse({
                'success': False,
                'error': 'You can only reply to reviews of your own questions.'
            }, status=403)
        
        # Get reply content
        comment = request.POST.get('comment', '').strip()
        
        if not comment:
            return JsonResponse({
                'success': False,
                'error': 'Reply comment cannot be empty.'
            }, status=400)
        
        try:
            # Create the reply
            reply = ReviewReply.objects.create(
                review=review,
                author=request.user.participant,
                comment=comment
            )
            
            # Return success with reply data
            return JsonResponse({
                'success': True,
                'reply': {
                    'id': reply.id,
                    'author': reply.author.name,
                    'comment': reply.comment,
                    'creation_time': reply.creation_time.strftime('%b %d, %Y at %I:%M %p')
                }
            })
            
        except Exception as e:
            return JsonResponse({
                'success': False,
                'error': 'Unable to submit reply. Please try again.'
            }, status=500)


class RecommendAcceptanceView(LoginRequiredMixin, View):
    """
    Recommend a question for acceptance (previously marked as needing revision).
    
    Only the reviewer who requested revision can recommend acceptance after the author replies.
    Note: This does NOT approve the question - only admins can do final approval.
    """
    
    def post(self, request, pk):
        """Handle recommendation for acceptance."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check if user has a participant profile
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile to recommend questions.')
            return redirect('questions:detail', pk=pk)
        
        # Check if question is under review
        if question.status.status != 'under_review':
            messages.error(request, '❌ Only questions under review can be recommended.')
            return redirect('questions:detail', pk=pk)
        
        # Find if this user has a "revise" review with a reply
        user_revise_reviews = QuestionReview.objects.filter(
            question=question,
            reviewer=request.user.participant,
            decision='revise'
        ).prefetch_related('replies')
        
        # Check if any of these reviews have replies
        can_recommend = False
        target_review = None
        for review in user_revise_reviews:
            if review.replies.exists() and not review.is_concluded:
                can_recommend = True
                target_review = review
                break
        
        if not can_recommend:
            messages.error(request, '❌ You can only recommend questions where you requested revision and the author has replied.')
            return redirect('questions:detail', pk=pk)
        
        try:
            # Update the review decision to 'accept' (which now means "Recommend for acceptance")
            # and mark as concluded
            target_review.decision = 'accept'
            target_review.is_concluded = True
            target_review.save()
            
            messages.success(request, f'✅ You have recommended "{question.title}" for acceptance. An admin will review your recommendation.')
            
        except Exception as e:
            messages.error(request, '❌ Unable to recommend question. Please try again.')
        
        return redirect('questions:detail', pk=pk)


class AdminApproveQuestionView(LoginRequiredMixin, View):
    """
    Admin-only view to approve a question for the benchmark.
    
    This is the final approval step after reviewers have recommended the question.
    """
    
    def post(self, request, pk):
        """Handle admin approval of a question."""
        question = get_object_or_404(Question, pk=pk)
        
        # Check if user has a participant profile and is admin
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile to approve questions.')
            return redirect('questions:detail', pk=pk)
        
        if not request.user.participant.is_admin:
            messages.error(request, '❌ Only administrators can approve questions.')
            return redirect('questions:detail', pk=pk)
        
        # Check if question is under review
        if question.status.status != 'under_review':
            messages.error(request, '❌ Only questions under review can be approved.')
            return redirect('questions:detail', pk=pk)
        
        # Check if there's at least one recommendation for acceptance
        has_recommendation = QuestionReview.objects.filter(
            question=question,
            decision='accept'
        ).exists()
        
        if not has_recommendation:
            messages.warning(request, 
                '⚠️ This question has not been recommended for acceptance by any reviewer. '
                'Are you sure you want to approve it?')
        
        try:
            # Update question status to approved
            approved_status = QuestionState.objects.get(status='approved')
            question.status = approved_status
            question.save()
            
            messages.success(request, 
                f'✅ Question "{question.title}" has been approved for the benchmark!')
            
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Unable to find approved status. Please contact system administrator.')
        except Exception as e:
            messages.error(request, f'❌ Unable to approve question: {str(e)}')
        
        return redirect('questions:detail', pk=pk)


class CloseReviewView(LoginRequiredMixin, View):
    """
    Close a review conversation without changing question status.
    
    Only the reviewer who made the review can close it after the author replies.
    This concludes the conversation and moves the review to "Closed Reviews".
    """
    
    def post(self, request, review_id):
        """Handle review closure."""
        review = get_object_or_404(QuestionReview, pk=review_id)
        
        # Check if user has a participant profile
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile to close reviews.')
            return redirect('questions:detail', pk=review.question.pk)
        
        # Check if user is the reviewer
        if review.reviewer != request.user.participant:
            messages.error(request, '❌ You can only close your own reviews.')
            return redirect('questions:detail', pk=review.question.pk)
        
        # Check if review has replies (can only close after author responds)
        if not review.replies.exists():
            messages.error(request, '❌ You can only close reviews after the author has replied.')
            return redirect('questions:detail', pk=review.question.pk)
        
        # Check if review is already concluded
        if review.is_concluded:
            messages.info(request, 'ℹ️ This review is already concluded.')
            return redirect('questions:detail', pk=review.question.pk)
        
        try:
            # Mark review as concluded
            review.is_concluded = True
            review.save()
            
            messages.success(request, f'✅ Review conversation concluded. The question remains under review.')
            
        except Exception as e:
            messages.error(request, '❌ Unable to close review. Please try again.')
        
        return redirect('questions:detail', pk=review.question.pk)


class RetractQuestionView(LoginRequiredMixin, View):
    """
    Retract a question from consideration for the benchmark.
    
    Sets question status to 'retracted' - less harsh than deletion but removes
    from active pool. Authors can later revert to draft to make changes.
    Only accessible to question authors and admins.
    """
    
    def post(self, request, pk):
        question = get_object_or_404(Question, pk=pk)
        
        # Permission check: author or admin only
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile.')
            return redirect('questions:detail', pk=pk)
        
        participant = request.user.participant
        is_author = question.author == participant
        is_admin = participant.is_admin
        
        if not (is_author or is_admin):
            messages.error(request, '❌ You can only retract your own questions or be an admin.')
            return redirect('questions:detail', pk=pk)
        
        # Check if question is in a retractable state (not already retracted)
        if question.status.status == 'retracted':
            messages.info(request, 'ℹ️ This question is already retracted.')
            return redirect('questions:detail', pk=pk)
        
        try:
            # Get retracted status
            retracted_status = QuestionState.objects.get(status='retracted')
            
            # Update question status
            question.status = retracted_status
            question.save()
            
            action_by = "by admin" if is_admin and not is_author else ""
            messages.success(request, f'✅ Question "{question.title}" has been retracted from benchmark consideration {action_by}. You can revert to draft status to make changes.')
            
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Retracted status not found in database.')
        except Exception as e:
            messages.error(request, '❌ Unable to retract question. Please try again.')
        
        return redirect('questions:detail', pk=pk)


class RevertToDraftFromRetractedView(LoginRequiredMixin, View):
    """
    Revert a retracted question back to draft status.
    
    Allows authors to resume working on questions that were previously retracted.
    Only accessible to question authors and admins.
    """
    
    def post(self, request, pk):
        question = get_object_or_404(Question, pk=pk)
        
        # Permission check: author or admin only
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile.')
            return redirect('questions:detail', pk=pk)
        
        participant = request.user.participant
        is_author = question.author == participant
        is_admin = participant.is_admin
        
        if not (is_author or is_admin):
            messages.error(request, '❌ You can only revert your own questions or be an admin.')
            return redirect('questions:detail', pk=pk)
        
        # Check if question is retracted
        if question.status.status != 'retracted':
            messages.info(request, 'ℹ️ This question is not in retracted status.')
            return redirect('questions:detail', pk=pk)
        
        try:
            # Get draft status
            draft_status = QuestionState.objects.get(status='draft')
            
            # Update question status
            question.status = draft_status
            question.save()
            
            action_by = "by admin" if is_admin and not is_author else ""
            messages.success(request, f'✅ Question "{question.title}" has been reverted to draft status {action_by}. You can now edit and resubmit for review.')
            
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Draft status not found in database.')
        except Exception as e:
            messages.error(request, '❌ Unable to revert question. Please try again.')
        
        return redirect('questions:detail', pk=pk)


class ActivateQuestionView(LoginRequiredMixin, View):
    """
    Activate an approved question for evaluation by all models.
    
    Changes question status from 'approved' to 'active' and creates evaluation
    queue entries for all active models in the system. Only admins can activate questions.
    """
    
    def post(self, request, pk):
        question = get_object_or_404(Question, pk=pk)
        
        # Permission check: admin only
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile.')
            return redirect('questions:detail', pk=pk)
        
        participant = request.user.participant
        if not participant.is_admin:
            messages.error(request, '❌ Only administrators can activate questions.')
            return redirect('questions:detail', pk=pk)
        
        # Check if question is in approved status
        if question.status.status != 'approved':
            messages.error(request, '❌ Only approved questions can be activated.')
            return redirect('questions:detail', pk=pk)
        
        try:
            from django.db import transaction
            from django.utils import timezone
            from model_evaluation.models import Model, ModelAttempt, EvaluationQueue
            
            with transaction.atomic():
                # 1. Update question status to active
                active_status = QuestionState.objects.get(status='active')
                question.status = active_status
                question.save()
                
                # 2. Get all active models
                active_models = Model.objects.filter(is_active=True)
                
                if not active_models.exists():
                    messages.warning(request, '⚠️ No active models found. Question activated but no evaluations queued.')
                    return redirect('questions:detail', pk=pk)
                
                # 3. Create evaluations for all active models
                evaluation_count = 0
                for model in active_models:
                    # Determine how many attempts to create based on model tier
                    max_attempts = 2 if model.tier.tier_number == 1 else 1
                    
                    for attempt_num in range(1, max_attempts + 1):
                        # Create model attempt
                        attempt = ModelAttempt.objects.create(
                            model=model,
                            question=question,
                            attempt_number=attempt_num,
                            time=timezone.now()
                        )
                        
                        # Create evaluation queue entry
                        EvaluationQueue.objects.create(
                            attempt=attempt,
                            status='pending',
                            submitted_at=timezone.now()
                        )
                        evaluation_count += 1
                
                # Success message with tier breakdown
                model_count = active_models.count()
                tier1_models = active_models.filter(tier__tier_number=1).count()
                
                # Calculate expected evaluations for clarity
                tier1_evaluations = tier1_models * 2  # 2 attempts each
                other_evaluations = (model_count - tier1_models) * 1  # 1 attempt each
                expected_total = tier1_evaluations + other_evaluations
                
                tier_info = ""
                if tier1_models > 0:
                    tier_info = f" (Tier 1 models get 2 attempts each)"
                
                messages.success(
                    request, 
                    f'✅ Question "{question.title}" has been activated! '
                    f'Queued {evaluation_count} evaluations for {model_count} model{"s" if model_count != 1 else ""}{tier_info}. '
                    f'Models will begin evaluation automatically.'
                )
                
        except QuestionState.DoesNotExist:
            messages.error(request, '❌ Active status not found in database.')
        except Exception as e:
            messages.error(request, '❌ Unable to activate question. Please try again.')

        return redirect('questions:detail', pk=pk)


class QuestionHistoryView(LoginRequiredMixin, DetailView):
    """
    Display version history for a question.

    Shows all versions of a question with creation timestamps, who made each change,
    and provides the ability to compare versions (diff view).

    Only the question author, reviewers, and admins can view the history.
    """
    model = Question
    template_name = 'questions/history.html'
    context_object_name = 'question'

    def get_object(self, queryset=None):
        """Get the question and check permissions."""
        question = super().get_object(queryset)

        # Check permissions: author, admin, or assigned reviewer
        user = self.request.user
        if not hasattr(user, 'participant'):
            raise PermissionDenied("You must have a participant profile.")

        is_author = question.author == user.participant
        is_admin = user.participant.is_admin
        is_reviewer = user.participant.role in ['reviewer', 'admin']

        if not (is_author or is_admin or is_reviewer):
            raise PermissionDenied("You don't have permission to view this question's history.")

        return question

    def get_context_data(self, **kwargs):
        """Add version history to context."""
        context = super().get_context_data(**kwargs)

        # Get all versions for this question
        versions = QuestionVersion.objects.filter(
            question=self.object
        ).select_related('created_by').order_by('-version_number')

        context['versions'] = versions

        # Add first and last versions for template use (querysets don't support negative indexing)
        if versions.exists():
            context['latest_version'] = versions.first()  # Highest version number (most recent)
            context['oldest_version'] = versions.last()   # Lowest version number (oldest)
        else:
            context['latest_version'] = None
            context['oldest_version'] = None

        # Check if we're comparing two versions
        v1 = self.request.GET.get('v1')
        v2 = self.request.GET.get('v2')

        if v1 and v2:
            try:
                version1 = QuestionVersion.objects.get(question=self.object, version_number=v1)
                version2 = QuestionVersion.objects.get(question=self.object, version_number=v2)
                context['comparing'] = True
                context['version1'] = version1
                context['version2'] = version2
                context['diff_data'] = self._generate_diff(version1, version2)
            except QuestionVersion.DoesNotExist:
                context['comparing'] = False
        else:
            context['comparing'] = False

        return context

    def _generate_diff(self, v1, v2):
        """Generate a diff between two versions."""
        import difflib

        diff_data = {}

        # Compare main text fields
        fields_to_compare = [
            ('title', 'Title'),
            ('text', 'Question Text'),
            ('solution', 'Solution'),
            ('tags', 'Tags'),
        ]

        for field_name, display_name in fields_to_compare:
            old_val = getattr(v1, field_name) or ''
            new_val = getattr(v2, field_name) or ''

            if old_val != new_val:
                # Generate unified diff
                old_lines = old_val.splitlines(keepends=True)
                new_lines = new_val.splitlines(keepends=True)
                diff = list(difflib.unified_diff(
                    old_lines, new_lines,
                    fromfile=f'Version {v1.version_number}',
                    tofile=f'Version {v2.version_number}',
                    lineterm=''
                ))
                diff_data[field_name] = {
                    'display_name': display_name,
                    'changed': True,
                    'diff_lines': diff,
                    'old_value': old_val,
                    'new_value': new_val,
                }
            else:
                diff_data[field_name] = {
                    'display_name': display_name,
                    'changed': False,
                }

        # Compare difficulty fields
        difficulty_fields = [
            ('difficulty_background', 'Background'),
            ('difficulty_reasoning', 'Reasoning'),
            ('difficulty_insight', 'Insight'),
            ('difficulty_compute', 'Compute'),
        ]

        for field_name, display_name in difficulty_fields:
            old_val = getattr(v1, field_name)
            new_val = getattr(v2, field_name)
            diff_data[field_name] = {
                'display_name': display_name,
                'changed': old_val != new_val,
                'old_value': old_val,
                'new_value': new_val,
            }

        # Compare subquestions
        v1_subquestions = v1.get_subquestions()
        v2_subquestions = v2.get_subquestions()

        diff_data['subquestions'] = {
            'display_name': 'Subquestions',
            'changed': v1_subquestions != v2_subquestions,
            'old_value': v1_subquestions,
            'new_value': v2_subquestions,
            'old_count': len(v1_subquestions),
            'new_count': len(v2_subquestions),
        }

        return diff_data


class PublishQuestionView(LoginRequiredMixin, View):
    """
    Publish an active question to make it publicly visible.

    Publishing a question makes it visible to all logged-in users, excludes it
    from benchmark results, and prevents grading of model answers from models
    released after publication. This action is IRREVERSIBLE. Only admins can publish questions.
    """

    def post(self, request, pk):
        question = get_object_or_404(Question, pk=pk)

        # Permission check: admin only
        if not hasattr(request.user, 'participant'):
            messages.error(request, '❌ You must have a participant profile.')
            return redirect('questions:detail', pk=pk)

        participant = request.user.participant
        if not participant.is_admin:
            messages.error(request, '❌ Only administrators can publish questions.')
            return redirect('questions:detail', pk=pk)

        # Check if question is active
        if question.status.status != 'active':
            messages.error(request, '❌ Only active questions can be published.')
            return redirect('questions:detail', pk=pk)

        # Check if question is already published
        if question.published:
            messages.warning(request, f'⚠️ This question is already published (on {question.published}).')
            return redirect('questions:detail', pk=pk)

        try:
            from django.utils import timezone

            # Set publication date to today
            question.published = timezone.now().date()
            question.save()

            messages.success(
                request,
                f'✅ Question "{question.title}" has been published! '
                f'It is now visible to all users and excluded from benchmark results.'
            )

        except Exception as e:
            messages.error(request, f'❌ Unable to publish question: {str(e)}')

        return redirect('questions:detail', pk=pk)


class EvaluationResultsView(LoginRequiredMixin, TemplateView):
    """
    Display evaluation results for a question with model performance table.

    Shows:
    - Model name
    - Subquestion score (points earned / total points)
    - Progress grade (from finalized grading sessions only)

    Access control:
    - Published questions: Everyone can view
    - Non-published questions: Only author or admin can view

    Special handling:
    - Models released after question publication date show "Not graded (released post-publication)"
    """
    template_name = 'questions/evaluation_results.html'

    def get(self, request, pk):
        question = get_object_or_404(Question, pk=pk)

        # Access control
        can_view = False
        if question.published:
            # Published questions are visible to all logged-in users
            can_view = True
        elif hasattr(request.user, 'participant'):
            # Non-published questions only visible to author or admin
            participant = request.user.participant
            can_view = (question.author == participant or participant.is_admin)

        if not can_view:
            messages.error(request, '❌ You do not have permission to view this question.')
            return redirect('questions:list')

        # Get all completed model evaluations for this question
        from model_evaluation.models import ModelAnswer, ModelAttempt, Model, ModelGrading, ModelGradingSession, GradingState, ModelSubquestionAnswer
        from questions.models import Subquestion

        # Get all model attempts with completed answers
        model_attempts = ModelAttempt.objects.filter(
            question=question
        ).select_related('model', 'model__tier', 'model__company').order_by(
            'model__tier__tier_number', 'model__model_name', 'attempt_number'
        )

        # Get finalized grading session for current user
        user_finalized_session = ModelGradingSession.objects.filter(
            question=question,
            grader=request.user,
            session_status='finalized'
        ).first()

        # Get any finalized grading session for displaying grades
        grading_session = ModelGradingSession.objects.filter(
            question=question,
            session_status='finalized'
        ).first()

        # Check if current user has an active (non-finalized) grading session
        grading_in_progress = False
        active_session = ModelGradingSession.objects.filter(
            question=question,
            grader=request.user
        ).exclude(session_status='finalized').first()
        if active_session:
            grading_in_progress = True

        # Get set of model_answer IDs in user's finalized session (for Tier 1 filtering)
        user_graded_answer_ids = set()
        if user_finalized_session:
            from model_evaluation.models import ModelGradingAlias
            user_graded_answer_ids = set(
                ModelGradingAlias.objects.filter(
                    session=user_finalized_session
                ).values_list('model_answer_id', flat=True)
            )

        # Group attempts by model and collect data
        from collections import defaultdict
        model_data = defaultdict(lambda: {
            'model': None,
            'tier': None,
            'attempts': [],
            'grade': None,
            'model_answer_ids': [],
            'latest_attempt_id': None
        })

        for attempt in model_attempts:
            # Check if there's a completed model answer
            try:
                model_answer = ModelAnswer.objects.get(attempt=attempt)
            except ModelAnswer.DoesNotExist:
                continue

            # Calculate subquestion score for this attempt
            subquestion_answers = ModelSubquestionAnswer.objects.filter(
                attempt=attempt
            )
            total_points = 0
            earned_points = 0
            for sq_answer in subquestion_answers:
                subquestion = sq_answer.subquestion
                total_points += subquestion.points if subquestion.points else 1

                # Use admin override if set, otherwise use is_correct
                if sq_answer.admin_override is not None:
                    is_correct = sq_answer.admin_override == 1
                else:
                    is_correct = sq_answer.is_correct == 1

                if is_correct:
                    earned_points += subquestion.points if subquestion.points else 1

            # Store attempt data
            model_key = attempt.model.id
            model_data[model_key]['model'] = attempt.model
            model_data[model_key]['tier'] = attempt.model.tier.tier_number
            model_data[model_key]['model_answer_ids'].append(model_answer.id)

            # Track latest attempt ID
            if model_data[model_key]['latest_attempt_id'] is None or attempt.id > model_data[model_key]['latest_attempt_id']:
                model_data[model_key]['latest_attempt_id'] = attempt.id

            model_data[model_key]['attempts'].append({
                'attempt_id': attempt.id,
                'attempt_number': attempt.attempt_number,
                'earned': earned_points,
                'total': total_points
            })

            # Get grade from finalized grading session (only if not already found)
            if model_data[model_key]['grade'] is None:
                grade_display = None
                try:
                    if grading_session and model_answer.released_for_grading:
                        # Get the grading for this model answer directly
                        grading = ModelGrading.objects.filter(
                            session=grading_session,
                            model_answer=model_answer,
                            grading_status='completed'
                        ).first()

                        if grading and grading.progress_grade is not None:
                            # Map progress grade to label
                            progress_labels = {
                                0: "0 - No progress",
                                1: "1 - Minimal progress",
                                2: "2 - Substantial progress",
                                3: "3 - Correct answer"
                            }
                            grade_display = progress_labels.get(grading.progress_grade, str(grading.progress_grade))

                        # Check if model was released after publication
                        if question.published and attempt.model.release_date:
                            if attempt.model.release_date > question.published:
                                grade_display = "Not graded (released post-publication)"

                except Exception:
                    pass

                if grade_display:
                    model_data[model_key]['grade'] = grade_display

        # Build final results with averaged scores
        results = []
        for model_key in sorted(model_data.keys(), key=lambda k: (model_data[k]['tier'], model_data[k]['model'].model_name)):
            data = model_data[model_key]

            # For Tier 1 and Tier 3 models: only show if user has graded them in a finalized session
            # Exception: Published questions show all models to everyone
            if not question.published and data['tier'] in [1, 3]:
                # Check if any of this model's answers are in the user's finalized session
                has_user_graded = any(answer_id in user_graded_answer_ids for answer_id in data['model_answer_ids'])
                if not has_user_graded:
                    continue

            # Calculate subquestion score as percentage
            if data['attempts']:
                if data['tier'] == 1:
                    # Tier 1: Group by attempt_number, take latest of each, then average
                    from collections import defaultdict
                    attempts_by_number = defaultdict(list)
                    for a in data['attempts']:
                        attempts_by_number[a['attempt_number']].append(a)

                    # Get latest attempt for each attempt_number
                    latest_attempts = []
                    for attempt_number, attempts in attempts_by_number.items():
                        latest = max(attempts, key=lambda x: x['attempt_id'])
                        latest_attempts.append(latest)

                    # Average across the latest attempts
                    avg_earned = sum(a['earned'] for a in latest_attempts) / len(latest_attempts)
                    avg_total = sum(a['total'] for a in latest_attempts) / len(latest_attempts)
                    if avg_total > 0:
                        percentage = (avg_earned / avg_total) * 100
                        subquestion_score = f"{percentage:.0f}%"
                    else:
                        subquestion_score = "N/A"
                else:
                    # Tier 2-4: Use only the most recent attempt
                    latest_attempt = [a for a in data['attempts'] if a['attempt_id'] == data['latest_attempt_id']][0]
                    if latest_attempt['total'] > 0:
                        percentage = (latest_attempt['earned'] / latest_attempt['total']) * 100
                        subquestion_score = f"{percentage:.0f}%"
                    else:
                        subquestion_score = "N/A"
            else:
                subquestion_score = "N/A"

            results.append({
                'model': data['model'],
                'tier': data['tier'],
                'subquestion_score': subquestion_score,
                'grade': data['grade'] or "N/A"
            })

        context = {
            'question': question,
            'results': results,
            'grading_in_progress': grading_in_progress,
            'has_finalized_session': user_finalized_session is not None
        }

        return self.render_to_response(context)


@login_required
@require_POST
def preview_markdown(request):
    """
    AJAX endpoint for rendering Markdown + LaTeX to HTML.

    Used by SimpleMDE editor for live preview.
    """
    try:
        data = json.loads(request.body)
        text = data.get('text', '')

        # Use the same markdown processor as template tags
        from .templatetags.question_extras import markdown_latex
        html = markdown_latex(text)

        return JsonResponse({'html': html})
    except Exception as e:
        return JsonResponse({'error': str(e)}, status=500)
