"""
Views for the review invitation system.

Provides interfaces for:
- Sending review invitations (admin only)
- Accepting/declining invitations (both internal and external reviewers)
- Managing invitation status (revoke, expire)
- External reviewer account creation and password setup
"""

from django.shortcuts import render, redirect, get_object_or_404
from django.contrib.auth.decorators import login_required
from django.contrib.auth.mixins import LoginRequiredMixin
from django.contrib import messages
from django.urls import reverse
from django.views import View
from django.views.generic import ListView, DetailView, FormView
from django.utils import timezone
from django.contrib.auth import get_user_model, login
from django.http import HttpResponseForbidden, HttpResponse
from django.utils.http import url_has_allowed_host_and_scheme
from django.db import models
from invitations.models import Invitation
from accounts.permissions import AdminRequiredMixin
from questions.models import Question
from .models import ReviewInvitation, ReviewAssignment, UserNotificationPreferences
from .forms import ReviewInvitationForm
from participants.models import Participant

User = get_user_model()


class SendReviewInvitationView(AdminRequiredMixin, FormView):
    """
    View for admins to send review invitations.
    """
    form_class = ReviewInvitationForm
    template_name = 'review_invitations/send_invitation.html'
    
    def dispatch(self, request, *args, **kwargs):
        self.question = get_object_or_404(Question, pk=kwargs['question_id'])
        return super().dispatch(request, *args, **kwargs)
    
    def get_form_kwargs(self):
        kwargs = super().get_form_kwargs()
        # Remove 'instance' if it exists (FormView doesn't pass it)
        kwargs.pop('instance', None)
        kwargs['question'] = self.question
        kwargs['inviter'] = self.request.user
        return kwargs
    
    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        context['question'] = self.question
        
        # Get inviter's name from Participant model
        from participants.models import Participant
        inviter_participant = Participant.objects.filter(user=self.request.user).first()
        context['inviter_name'] = inviter_participant.name if inviter_participant else (self.request.user.get_full_name() or self.request.user.email)
        
        # Pre-fill email template
        if not self.request.POST:
            form = context['form']
            form.initial['custom_message'] = form.get_email_template(is_external=False)
        
        return context
    
    def form_valid(self, form):
        invitation_type = form.cleaned_data['invitation_type']
        send_email = form.cleaned_data.get('send_email', True)
        
        if invitation_type == 'existing':
            # Handle existing user invitation
            user = form.cleaned_data['existing_user']
            
            if not send_email:
                # Direct assignment without invitation email (regardless of whether user has email)
                # Create ReviewInvitation with accepted status
                review_invitation = ReviewInvitation.objects.create(
                    invitation=None,  # No invitation object needed
                    question=self.question,
                    invited_by=self.request.user,
                    custom_message=form.cleaned_data.get('custom_message', ''),
                    status='accepted'  # Direct assignment = already accepted
                )
                
                # Create review assignment immediately
                ReviewAssignment.objects.get_or_create(
                    user=user,
                    question=self.question,
                    defaults={
                        'invitation': review_invitation,
                        'is_active': True,
                    }
                )
                
                messages.success(
                    self.request,
                    f'{user.get_full_name() or user.email} has been assigned to review this question'
                )
                
            else:
                # Create invitation with email for internal reviewer
                email = user.email
                
                if not email:
                    messages.error(
                        self.request,
                        f'Cannot send email invitation: {user.get_full_name()} has no email address. '
                        'Please use direct assignment (uncheck "Send email notification").'
                    )
                    return redirect(self.request.path)
                
                # Check if this specific user already has ANY assignment for this question
                # This check is redundant with form validation but provides safety
                existing_assignment = ReviewAssignment.objects.filter(
                    user=user,
                    question=self.question
                ).first()
                
                if existing_assignment:
                    participant_name = user.participant.name if hasattr(user, "participant") else email
                    if existing_assignment.is_active:
                        messages.warning(
                            self.request,
                            f'{participant_name} has already accepted the invitation to review this question.'
                        )
                    else:
                        messages.warning(
                            self.request,
                            f'{participant_name} has already been invited to review this question.'
                        )
                    return redirect(self.question.get_absolute_url())
                
                # For internal reviewers, create ReviewInvitation with custom token (no django-invitation)
                review_invitation = ReviewInvitation.objects.create(
                    invitation=None,  # No django-invitation for internal reviewers
                    token=ReviewInvitation.generate_token(),  # Custom token
                    question=self.question,
                    invited_by=self.request.user,
                    custom_message=form.cleaned_data.get('custom_message', ''),
                    status='pending'
                )
                
                # Store the intended reviewer in ReviewAssignment but mark as not active yet
                ReviewAssignment.objects.create(
                    user=user,
                    question=self.question,
                    invitation=review_invitation,
                    is_active=False  # Will become active when accepted
                )
                
                # Send the invitation email
                self.send_invitation_email(review_invitation)
                messages.success(
                    self.request,
                    f'Review invitation sent to {email}'
                )
            
        else:  # external
            # Handle external reviewer invitation
            first_name = form.cleaned_data['external_first_name']
            last_name = form.cleaned_data['external_last_name']
            email = form.cleaned_data['external_email']
            
            # Check if this external reviewer has already been invited to ANY question
            existing_invitation = Invitation.objects.filter(email=email).first()
            if existing_invitation:
                # Check if they accepted and now have an account
                from django.contrib.auth import get_user_model
                User = get_user_model()
                if User.objects.filter(email=email).exists():
                    messages.error(
                        self.request,
                        f'{email} already has an account. Please use "Select existing reviewer" instead.'
                    )
                else:
                    messages.error(
                        self.request,
                        f'{email} has already been invited to review. External reviewers can only be invited once.'
                    )
                return redirect(self.question.get_absolute_url())
            
            # Create new django-invitation for external reviewer
            invitation = Invitation.create(
                email=email,
                inviter=self.request.user
            )
            invitation.sent = timezone.now()
            invitation.save()
            
            # Create our ReviewInvitation
            review_invitation = ReviewInvitation.objects.create(
                invitation=invitation,
                question=self.question,
                invited_by=self.request.user,
                external_first_name=first_name,
                external_last_name=last_name,
                custom_message=form.cleaned_data.get('custom_message', ''),
                status='pending'
            )
            
            # Send the invitation email (always for external)
            self.send_invitation_email(review_invitation)
            messages.success(
                self.request,
                f'Review invitation sent to {email}'
            )
        
        return redirect(self.question.get_absolute_url())
    
    def send_invitation_email(self, review_invitation):
        """
        Send the invitation email with custom context.
        """
        from django.urls import reverse
        from django.core.mail import send_mail
        from django.contrib.auth import get_user_model
        from participants.models import Participant
        
        User = get_user_model()
        is_external = bool(review_invitation.external_first_name)
        
        # Determine reviewer name
        if is_external:
            reviewer_name = f"{review_invitation.external_first_name} {review_invitation.external_last_name}"
            email = review_invitation.invitation.email
        else:
            # Internal reviewer - get user info
            # Find the user from the review assignment or by matching question/inviter context
            assignment = review_invitation.assignments.first()
            if assignment:
                user = assignment.user
            else:
                # This shouldn't happen, but handle gracefully
                user = None
            
            if user:
                participant = Participant.objects.filter(user=user).first()
                # Try participant name first, then user's full name, then first name only
                if participant and participant.name:
                    reviewer_name = participant.name
                elif user.first_name:
                    reviewer_name = user.get_full_name() if user.last_name else user.first_name
                else:
                    # Last resort - use email prefix
                    reviewer_name = user.email.split('@')[0].replace('.', ' ').title()
                email = user.email
            else:
                reviewer_name = "Reviewer"
                email = None
        
        # Build accept and decline URLs based on type
        if review_invitation.token:
            # Internal reviewer - use custom token URLs
            accept_url = self.request.build_absolute_uri(
                reverse('review_invitations:accept_internal', kwargs={'token': review_invitation.token})
            )
            decline_url = self.request.build_absolute_uri(
                reverse('review_invitations:decline_internal', kwargs={'token': review_invitation.token})
            )
        elif review_invitation.invitation:
            # External reviewer - use django-invitations URLs
            accept_url = self.request.build_absolute_uri(
                reverse('review_invitations:accept', kwargs={'key': review_invitation.invitation.key})
            )
            decline_url = self.request.build_absolute_uri(
                reverse('review_invitations:decline', kwargs={'key': review_invitation.invitation.key})
            )
        else:
            # This shouldn't happen
            return
        
        # Get inviter's name
        inviter_participant = Participant.objects.filter(user=self.request.user).first()
        inviter_name = inviter_participant.name if inviter_participant else (self.request.user.get_full_name() or self.request.user.email)
        
        # Prepare email content
        email_body = review_invitation.custom_message
        if email_body:
            email_body = email_body.replace('[ACCEPT_URL]', accept_url)
            email_body = email_body.replace('[DECLINE_URL]', decline_url)
        else:
            # Default message if no custom message provided
            email_body = f"""Dear {reviewer_name},

You have been invited to review the question "{review_invitation.question.title}".

To accept this invitation, please click here: {accept_url}

To decline this invitation, please click here: {decline_url}

Best regards,
{inviter_name}"""
        
        # Send email with proper headers
        if email:
            from django.conf import settings
            from django.core.mail import EmailMessage
            
            # Get inviter's participant name for the display name
            inviter_participant = Participant.objects.filter(user=self.request.user).first()
            inviter_display_name = inviter_participant.name if inviter_participant else self.request.user.get_full_name()
            if not inviter_display_name:
                inviter_display_name = "IMProofBench Team"
            
            # Format the from email with display name
            # Use the system email but with the inviter's name
            from_email = f'"{inviter_display_name}" <{settings.DEFAULT_FROM_EMAIL}>'
            
            # Create email message
            email_message = EmailMessage(
                subject=f'Review Request: {review_invitation.question.title}',
                body=email_body,
                from_email=from_email,
                to=[email],
            )
            
            # Add Reply-To header with inviter's actual email
            if self.request.user.email:
                # Format reply-to with name
                reply_to_email = f'"{inviter_display_name}" <{self.request.user.email}>'
                email_message.reply_to = [reply_to_email]
            
            # Send the email
            email_message.send(fail_silently=False)


class AcceptInvitationView(View):
    """
    Handle accepting a review invitation.
    """
    
    def get(self, request, key):
        # Import User at the beginning
        from django.contrib.auth import get_user_model
        User = get_user_model()
        
        # Get the invitation
        invitation = get_object_or_404(Invitation, key=key)
        
        # Check if expired
        if invitation.key_expired():
            messages.error(request, 
                'This invitation has expired. Please contact an administrator '
                'if you still wish to review this question.'
            )
            return redirect('home')
        
        # Get the review invitation
        review_invitation = getattr(invitation, 'review_invitation', None)
        if not review_invitation:
            messages.error(request, 'Invalid invitation.')
            return redirect('home')
        
        # Mark as accessed
        review_invitation.mark_accessed()
        
        # Check if invitation has been revoked
        if review_invitation.status == 'revoked':
            messages.error(request, 
                'This invitation has been revoked and is no longer valid.')
            return redirect('home')
        
        # Check if invitation was declined
        if review_invitation.status == 'declined':
            messages.error(request, 
                'This invitation has been declined and cannot be accepted. '
                'Please contact an administrator if you wish to reconsider.')
            return redirect('home')
        
        # Check if already accepted
        if review_invitation.status == 'accepted':
            # Already accepted - handle re-access for external reviewers
            if review_invitation.external_first_name:
                # External reviewer - automatically log them in again
                try:
                    user = User.objects.get(email=invitation.email)
                    # Log them in automatically
                    login(request, user, backend='django.contrib.auth.backends.ModelBackend')
                    messages.info(request, 'Welcome back! You have been automatically logged in.')
                    return redirect(review_invitation.question.get_absolute_url())
                except User.DoesNotExist:
                    messages.error(request, 'User account not found. Please contact support.')
                    return redirect('home')
            else:
                # Internal reviewer - they should already be authenticated
                if request.user.is_authenticated:
                    return redirect(review_invitation.question.get_absolute_url())
                else:
                    messages.info(request, 'Please log in to view the question.')
                    return redirect('account_login')
        
        # Accept the invitation
        review_invitation.status = 'accepted'
        review_invitation.save()
        
        # Handle external reviewer account creation
        if review_invitation.external_first_name:
            # Create or get user
            user, created = User.objects.get_or_create(
                email=invitation.email,
                defaults={
                    'first_name': review_invitation.external_first_name,
                    'last_name': review_invitation.external_last_name,
                }
            )
            
            if created:
                # Set unusable password for now
                user.set_unusable_password()
                user.save()
                
                # Create participant
                Participant.objects.get_or_create(
                    user=user,
                    defaults={
                        'name': f"{user.first_name} {user.last_name}",
                        'email': user.email,
                        'role': Participant.ROLE_CONTRIBUTOR,
                        'is_active': True,
                    }
                )
                
                # Create notification preferences
                UserNotificationPreferences.objects.get_or_create(
                    user=user,
                    defaults={
                        'show_in_leaderboard': False,
                        'notify_reviewed_question_grading': False,
                    }
                )
                
                review_invitation.created_account = user
                review_invitation.save()
            
            # Log them in
            login(request, user, backend='django.contrib.auth.backends.ModelBackend')
            
        else:
            # Existing user - get user by email
            try:
                user = User.objects.get(email=invitation.email)
            except User.DoesNotExist:
                messages.error(request, 'User account not found.')
                return redirect('home')
            
            if not request.user.is_authenticated:
                messages.info(request, 'Please log in to continue.')
                return redirect('account_login')
            elif request.user != user:
                messages.error(request, 'This invitation was sent to a different user.')
                return redirect('home')
        
        # Create review assignment
        ReviewAssignment.objects.get_or_create(
            user=user,
            question=review_invitation.question,
            defaults={
                'invitation': review_invitation,
                'is_active': True,
            }
        )
        
        # Mark invitation as accepted in django-invitations
        invitation.accepted = True
        invitation.save()
        
        messages.success(request, 
            f'Thank you for accepting the invitation to review "{review_invitation.question.title}"'
        )
        
        # Redirect to the question review page
        return redirect('questions:review', pk=review_invitation.question.pk)


class DeclineInvitationView(View):
    """
    Handle declining a review invitation.
    """
    
    def get(self, request, key):
        # Get the invitation
        invitation = get_object_or_404(Invitation, key=key)
        
        # Get the review invitation
        review_invitation = getattr(invitation, 'review_invitation', None)
        if not review_invitation:
            messages.error(request, 'Invalid invitation.')
            return redirect('home')
        
        # Mark as accessed
        review_invitation.mark_accessed()
        
        # Check if invitation has been revoked
        if review_invitation.status == 'revoked':
            messages.error(request, 
                'This invitation has been revoked and is no longer valid.')
            return redirect('home')
        
        # Check if already processed
        if review_invitation.status in ['accepted', 'declined']:
            messages.info(request, 
                f'This invitation has already been {review_invitation.status}.')
            return redirect('home')
        
        # Decline the invitation
        review_invitation.status = 'declined'
        review_invitation.save()
        
        return render(request, 'review_invitations/declined.html', {
            'question_title': review_invitation.question.title
        })


class ReviewDashboardView(AdminRequiredMixin, ListView):
    """
    Dashboard for managing review invitations.
    """
    model = ReviewInvitation
    template_name = 'review_invitations/dashboard.html'
    context_object_name = 'invitations'
    paginate_by = 20
    
    def get_queryset(self):
        queryset = super().get_queryset()
        
        # Filter by status if requested
        status = self.request.GET.get('status')
        if status == 'reviewed':
            # Special case: filter for accepted invitations that have submitted reviews
            # We'll filter this in memory after fetching accepted invitations
            queryset = queryset.filter(status='accepted')
        elif status:
            queryset = queryset.filter(status=status)
        
        return queryset.select_related(
            'question', 'invited_by', 'invitation__inviter', 'created_account'
        ).prefetch_related('assignments').order_by('-created_at')
    
    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        from django.contrib.auth import get_user_model
        from participants.models import Participant
        User = get_user_model()
        
        # Count reviewed invitations first (accepted + review submitted)
        from questions.models import QuestionReview
        reviewed_count = 0
        accepted_total = 0
        
        for invitation in ReviewInvitation.objects.filter(status='accepted'):
            accepted_total += 1
            reviewer_user = None
            assignment = invitation.assignments.first()
            if assignment:
                reviewer_user = assignment.user
            elif invitation.created_account:
                reviewer_user = invitation.created_account
            
            if reviewer_user:
                participant = Participant.objects.filter(user=reviewer_user).first()
                if participant:
                    review_exists = QuestionReview.objects.filter(
                        question=invitation.question,
                        reviewer=participant
                    ).exists()
                    if review_exists:
                        reviewed_count += 1
        
        # Get counts by status
        context['status_counts'] = {
            'pending': ReviewInvitation.objects.filter(status='pending').count(),
            'accepted': accepted_total - reviewed_count,  # Only count unreviewed acceptances
            'declined': ReviewInvitation.objects.filter(status='declined').count(),
            'expired': ReviewInvitation.objects.filter(status='expired').count(),
            'revoked': ReviewInvitation.objects.filter(status='revoked').count(),
            'reviewed': reviewed_count,
        }
        
        # Get total count (unfiltered) for the "All" display
        context['total_count'] = ReviewInvitation.objects.count()
        
        # Get invitations needing reminders
        context['needs_reminder'] = []
        for invitation in ReviewInvitation.objects.filter(status='pending').select_related('invitation', 'invited_by'):
            if invitation.needs_reminder:
                # Add invited user info for internal invitations
                if invitation.invitation and not invitation.external_first_name:
                    try:
                        invited_user = User.objects.get(email=invitation.invitation.email)
                        invitation.invited_user = invited_user
                        # Also get participant for name
                        participant = Participant.objects.filter(user=invited_user).first()
                        if participant:
                            invitation.invited_participant = participant
                    except User.DoesNotExist:
                        pass
                context['needs_reminder'].append(invitation)
        
        # Add user/participant info to the main queryset items
        from questions.models import QuestionReview
        filtered_invitations = []
        status_filter = self.request.GET.get('status')
        
        for invitation in context['invitations']:
            # For direct assignments (no invitation object), get user from assignment
            if not invitation.invitation and not invitation.external_first_name:
                # This is a direct assignment - get the user from the assignment
                assignment = invitation.assignments.first()
                if assignment:
                    invitation.invited_user = assignment.user
                    participant = Participant.objects.filter(user=assignment.user).first()
                    if participant:
                        invitation.invited_participant = participant
            # For email invitations to internal users
            elif invitation.invitation and not invitation.external_first_name and not invitation.created_account:
                try:
                    invited_user = User.objects.get(email=invitation.invitation.email)
                    invitation.invited_user = invited_user
                    # Also get participant for name
                    participant = Participant.objects.filter(user=invited_user).first()
                    if participant:
                        invitation.invited_participant = participant
                except User.DoesNotExist:
                    pass
            
            # Add inviter participant info for better name display
            if invitation.invited_by:
                inviter_participant = Participant.objects.filter(user=invitation.invited_by).first()
                if inviter_participant:
                    invitation.inviter_participant = inviter_participant
            
            # Check if review has been submitted (for accepted invitations)
            invitation.review_submitted = False
            if invitation.status == 'accepted':
                # Get the reviewer user
                reviewer_user = None
                assignment = invitation.assignments.first()
                if assignment:
                    reviewer_user = assignment.user
                elif invitation.created_account:
                    reviewer_user = invitation.created_account
                
                # Check if this user submitted a review for this question
                if reviewer_user:
                    participant = Participant.objects.filter(user=reviewer_user).first()
                    if participant:
                        review_exists = QuestionReview.objects.filter(
                            question=invitation.question,
                            reviewer=participant
                        ).exists()
                        invitation.review_submitted = review_exists
            
            # Filter for reviewed status if requested
            if status_filter == 'reviewed':
                # Only show accepted invitations that have been reviewed
                if invitation.status == 'accepted' and invitation.review_submitted:
                    filtered_invitations.append(invitation)
            elif status_filter == 'accepted':
                # For accepted filter, show only those NOT reviewed yet
                if invitation.status == 'accepted' and not invitation.review_submitted:
                    filtered_invitations.append(invitation)
            else:
                # For other filters or no filter, include all
                filtered_invitations.append(invitation)
        
        # Replace the invitations list if we're filtering for reviewed or accepted
        if status_filter in ['reviewed', 'accepted']:
            context['invitations'] = filtered_invitations
        
        return context


class AcceptInternalInvitationView(LoginRequiredMixin, View):
    """
    Handle accepting a review invitation for internal reviewers (token-based).
    """
    
    def get(self, request, token):
        # Get the review invitation by token
        review_invitation = get_object_or_404(ReviewInvitation, token=token)
        
        # Mark as accessed
        review_invitation.mark_accessed()
        
        # Check if invitation has been revoked
        if review_invitation.status == 'revoked':
            messages.error(request, 
                'This invitation has been revoked and is no longer valid.')
            return redirect('home')
        
        # Check if invitation was declined
        if review_invitation.status == 'declined':
            messages.error(request, 
                'This invitation has been declined and cannot be accepted. '
                'Please contact an administrator if you wish to reconsider.')
            return redirect('home')
        
        # Check if already accepted
        if review_invitation.status == 'accepted':
            messages.info(request, 'You have already accepted this invitation.')
            return redirect(review_invitation.question.get_absolute_url())
        
        # Accept the invitation
        review_invitation.status = 'accepted'
        review_invitation.save()
        
        # Activate the review assignment
        assignment = ReviewAssignment.objects.filter(
            invitation=review_invitation
        ).first()
        
        if assignment:
            # Verify this is the right user
            if assignment.user != request.user:
                messages.error(request, 'This invitation was sent to a different user.')
                return redirect('home')
            assignment.is_active = True
            assignment.save()
        else:
            # Shouldn't happen, but handle gracefully
            ReviewAssignment.objects.get_or_create(
                user=request.user,
                question=review_invitation.question,
                defaults={
                    'invitation': review_invitation,
                    'is_active': True,
                }
            )
        
        messages.success(request, 
            f'Thank you for accepting the invitation to review "{review_invitation.question.title}"'
        )
        
        # Redirect to the question review page
        return redirect('questions:review', pk=review_invitation.question.pk)


class DeclineInternalInvitationView(View):
    """
    Handle declining a review invitation for internal reviewers (token-based).
    """
    
    def get(self, request, token):
        # Get the review invitation by token
        review_invitation = get_object_or_404(ReviewInvitation, token=token)
        
        # Mark as accessed
        review_invitation.mark_accessed()
        
        # Check if invitation has been revoked
        if review_invitation.status == 'revoked':
            messages.error(request, 
                'This invitation has been revoked and is no longer valid.')
            return redirect('home')
        
        # Check if already processed
        if review_invitation.status in ['accepted', 'declined']:
            messages.info(request, 
                f'This invitation has already been {review_invitation.status}.')
            return redirect('home')
        
        # Decline the invitation
        review_invitation.status = 'declined'
        review_invitation.save()
        
        return render(request, 'review_invitations/declined.html', {
            'question_title': review_invitation.question.title
        })


class SetPasswordView(LoginRequiredMixin, View):
    """
    Allow external reviewers to set a password after submitting a review.
    """
    template_name = 'review_invitations/set_password.html'
    
    def get(self, request):
        # Check if user needs to set password
        if request.user.has_usable_password():
            messages.info(request, 'You already have a password set.')
            return redirect('home')
        
        return render(request, self.template_name)
    
    def post(self, request):
        password1 = request.POST.get('password1')
        password2 = request.POST.get('password2')
        
        if password1 != password2:
            messages.error(request, 'Passwords do not match.')
            return render(request, self.template_name)
        
        if len(password1) < 8:
            messages.error(request, 'Password must be at least 8 characters long.')
            return render(request, self.template_name)
        
        # Set the password
        request.user.set_password(password1)
        request.user.save()
        
        # Re-authenticate to maintain session
        login(request, request.user, backend='django.contrib.auth.backends.ModelBackend')
        
        messages.success(request, 'Password set successfully!')
        return redirect('home')


class RevokeInvitationView(AdminRequiredMixin, View):
    """
    Revoke a pending review invitation.
    """
    
    def post(self, request, invitation_id):
        invitation = get_object_or_404(ReviewInvitation, pk=invitation_id)
        
        # Check if invitation can be revoked
        if invitation.status != 'pending':
            messages.warning(request, 
                f'Cannot revoke invitation with status "{invitation.get_status_display()}".')
            return redirect('review_invitations:dashboard')
        
        # Update invitation status
        invitation.status = 'revoked'
        invitation.save()
        
        # Deactivate any associated ReviewAssignment
        ReviewAssignment.objects.filter(invitation=invitation).update(is_active=False)
        
        # Get reviewer name for message
        if invitation.external_first_name:
            reviewer_name = f"{invitation.external_first_name} {invitation.external_last_name}"
        else:
            # Find the user from assignment
            assignment = invitation.assignments.first()
            if assignment:
                reviewer_name = assignment.user.get_full_name() or assignment.user.email
            else:
                reviewer_name = "the reviewer"
        
        messages.success(request, 
            f'Review invitation to {reviewer_name} for "{invitation.question.title}" has been revoked.')
        
        # Return to dashboard or referrer (validated)
        referer = request.META.get('HTTP_REFERER')
        if referer and url_has_allowed_host_and_scheme(referer, allowed_hosts={request.get_host()}):
            return redirect(referer)
        return redirect('review_invitations:dashboard')


class DeclineAfterAcceptView(LoginRequiredMixin, View):
    """
    Allow a reviewer to decline a review request after initially accepting it.
    This is for users who accepted but then viewed the full question and decided not to review.
    """
    
    def post(self, request, question_id):
        from questions.models import Question
        question = get_object_or_404(Question, pk=question_id)
        
        # Find the user's accepted invitation for this question
        review_invitation = ReviewInvitation.objects.filter(
            assignments__user=request.user,
            assignments__question=question,
            assignments__is_active=True,
            status='accepted'
        ).first()
        
        if not review_invitation:
            messages.error(request, 'You do not have an active review invitation for this question.')
            return redirect('questions:detail', pk=question_id)
        
        # Update invitation status to declined
        review_invitation.status = 'declined'
        review_invitation.save()
        
        # Deactivate the review assignment
        ReviewAssignment.objects.filter(
            user=request.user,
            question=question,
            invitation=review_invitation
        ).update(is_active=False)
        
        messages.info(request, 
            f'You have declined the review request for "{question.title}". '
            'The question has been removed from your review assignments.'
        )
        
        # Redirect to questions list instead of the question detail
        # (since they no longer have access to review it)
        return redirect('questions:list')
