from django import forms
from django.contrib.auth import get_user_model
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Layout, Field, Row, Column, HTML, Div, Submit
from .models import ReviewInvitation
from questions.models import Question

User = get_user_model()


class ReviewInvitationForm(forms.Form):
    """
    Form for inviting reviewers to review a question.
    """
    
    # Radio choice for invitation type
    invitation_type = forms.ChoiceField(
        choices=[
            ('existing', 'Existing User'),
            ('external', 'External Reviewer'),
        ],
        initial='existing',
        widget=forms.RadioSelect,
        label='Reviewer Type'
    )
    
    # For existing users
    existing_user = forms.ModelChoiceField(
        queryset=User.objects.filter(is_active=True).order_by('first_name', 'last_name', 'email'),
        required=False,
        label='Select User',
        help_text='Choose an existing user from the system',
        widget=forms.Select(attrs={'class': 'form-select'})
    )
    
    # For external reviewers
    external_first_name = forms.CharField(
        max_length=100,
        required=False,
        label='First Name'
    )
    
    external_last_name = forms.CharField(
        max_length=100,
        required=False,
        label='Last Name'
    )
    
    external_email = forms.EmailField(
        required=False,
        label='Email Address',
        help_text='Email address of the external reviewer'
    )
    
    # Email notification settings
    send_email = forms.BooleanField(
        initial=True,
        required=False,
        label='Send email notification',
        help_text='Uncheck to create invitation without sending email'
    )
    
    # Custom email message  
    custom_message = forms.CharField(
        widget=forms.Textarea(attrs={'rows': 15}),
        required=False,
        label='Email Message',
        help_text='Edit the invitation email as needed. Keep [ACCEPT_URL] and [DECLINE_URL] placeholders intact - they will be replaced with actual links when sending.'
    )
    
    def __init__(self, *args, **kwargs):
        self.question = kwargs.pop('question', None)
        self.inviter = kwargs.pop('inviter', None)
        super().__init__(*args, **kwargs)
        
        # Customize how users are displayed in the dropdown
        from participants.models import Participant
        self.fields['existing_user'].label_from_instance = lambda obj: self._get_user_display_name(obj)
        
        # Exclude the question author from the list of possible reviewers
        if self.question and self.question.author:
            # Get the user associated with the author
            author_user = User.objects.filter(participant=self.question.author).first()
            if author_user:
                self.fields['existing_user'].queryset = User.objects.filter(
                    is_active=True
                ).exclude(
                    id=author_user.id
                ).order_by('first_name', 'last_name', 'email')
        
        # Set up crispy forms helper
        self.helper = FormHelper()
        self.helper.form_id = 'review-invitation-form'
        self.helper.form_method = 'post'
        
        self.helper.layout = Layout(
            HTML('<div class="mb-4">'),
            HTML('<h5>Question Information</h5>'),
            HTML(f'<p><strong>Title:</strong> {self.question.title}</p>'),
            HTML(f'<p><strong>Author:</strong> {self.question.author.name if self.question.author else "Unknown"}</p>'),
            HTML(f'<a href="{self.question.get_absolute_url()}" target="_blank">View Question Details</a>'),
            HTML('</div>'),
            
            HTML('<hr class="my-4">'),
            
            Field('invitation_type', css_class='reviewer-type-radio'),
            
            # Existing user section
            Div(
                Field('existing_user'),
                css_class='existing-user-section',
                css_id='existing-user-section'
            ),
            
            # External reviewer section
            Div(
                Row(
                    Column('external_first_name', css_class='col-md-6'),
                    Column('external_last_name', css_class='col-md-6'),
                ),
                Field('external_email'),
                HTML('<div id="email-collision-warning" class="alert alert-warning d-none"></div>'),
                css_class='external-reviewer-section d-none',
                css_id='external-reviewer-section'
            ),
            
            HTML('<hr class="my-4">'),
            
            # Email settings
            Field('send_email'),
            Div(
                Field('custom_message'),
                css_id='email-message-section'
            ),
            
            HTML('<div class="mt-4">'),
            Submit('submit', 'Send Invitation', css_class='btn btn-primary'),
            HTML(f'<a href="{self.question.get_absolute_url()}" class="btn btn-secondary ms-2">Cancel</a>'),
            HTML('</div>')
        )
    
    def _get_user_display_name(self, user):
        """Get the display name for a user in the dropdown."""
        from participants.models import Participant
        participant = Participant.objects.filter(user=user).first()
        if participant and participant.name:
            return f"{participant.name} ({user.email})"
        elif user.first_name or user.last_name:
            name = user.get_full_name()
            return f"{name} ({user.email})"
        else:
            return user.email
    
    def clean(self):
        cleaned_data = super().clean()
        invitation_type = cleaned_data.get('invitation_type')
        
        if invitation_type == 'existing':
            existing_user = cleaned_data.get('existing_user')
            if not existing_user:
                raise forms.ValidationError('Please select an existing user.')
                
            # Check if this user is already assigned to review this question
            # Check for ANY assignment (active or pending invitation)
            from .models import ReviewAssignment
            existing_assignment = ReviewAssignment.objects.filter(
                user=existing_user,
                question=self.question
            ).first()
            
            if existing_assignment:
                # Provide specific message based on status
                if existing_assignment.is_active:
                    raise forms.ValidationError(
                        f'{existing_user.get_full_name() or existing_user.email} '
                        f'has already accepted the invitation to review this question.'
                    )
                else:
                    raise forms.ValidationError(
                        f'{existing_user.get_full_name() or existing_user.email} '
                        f'has already been invited to review this question (invitation is pending).'
                    )
                
        elif invitation_type == 'external':
            first_name = cleaned_data.get('external_first_name')
            last_name = cleaned_data.get('external_last_name')
            email = cleaned_data.get('external_email')
            
            if not first_name:
                raise forms.ValidationError('First name is required for external reviewers.')
            if not last_name:
                raise forms.ValidationError('Last name is required for external reviewers.')
            if not email:
                raise forms.ValidationError('Email address is required for external reviewers.')
                
            # Check if email already exists
            if User.objects.filter(email=email).exists():
                user = User.objects.get(email=email)
                self.add_error('external_email', 
                    f'This email already exists in the system for user: '
                    f'{user.get_full_name() or user.email} (ID: {user.id}). '
                    f'Please use "Existing User" option instead.'
                )
        
        # For external reviewers, email must be sent (force it to True)
        if invitation_type == 'external':
            cleaned_data['send_email'] = True  # Always True for external reviewers
        
        # Validate that placeholders are present in the custom message if sending email
        if cleaned_data.get('send_email'):
            custom_msg = cleaned_data.get('custom_message', '')
            if custom_msg and ('[ACCEPT_URL]' not in custom_msg or '[DECLINE_URL]' not in custom_msg):
                self.add_error('custom_message', 
                    'The email message must contain both [ACCEPT_URL] and [DECLINE_URL] placeholders. '
                    'These will be replaced with actual links when the email is sent.'
                )
        
        return cleaned_data
    
    def get_inviter_name(self):
        """
        Get the human-readable name of the inviter from Participant model.
        """
        if not self.inviter:
            return 'The IMProofBench Team'
        
        from participants.models import Participant
        participant = Participant.objects.filter(user=self.inviter).first()
        if participant and participant.name:
            return participant.name
        return self.inviter.get_full_name() or self.inviter.email
    
    def get_email_template(self, is_external=False):
        """
        Generate the email template based on reviewer type.
        """
        reviewer_name = "{{reviewer_first_name}}"
        
        if is_external:
            intro = f"""
My name is {self.get_inviter_name()} and I am part of a small team of mathematicians at <redacted>
and <redacted>, studying the question of how good today's AI models are at solving 
research-level math questions. As part of this IMProofBench project, we are building a collection 
of challenging mathematical problems to use for testing the AI performance. 

We would like to ask for your help in verifying the mathematical correctness of one such question. 
If you are interested to learn more about the project, further information is available at 
https://improofbench.math.ethz.ch/faq/"""
        else:
            intro = ""
            
        template = f"""Dear {reviewer_name},

{intro}

The following question was submitted for inclusion in the IMProofBench dataset:

Title: {self.question.title}
Author: {self.question.author.name if self.question.author else "Unknown"}

Would you be willing to review this question and:
- Verify that the phrasing is well-defined and unambiguous
- Confirm the provided solution is mathematically correct
- Make any suggestions for improvements (e.g., additional unique-answer subquestions)

We estimate that for most problems this should take between 10 and 30 minutes. 

You can view the full submitted problem and write a review at:
[ACCEPT_URL]

There you will also have the option to decline this review request after viewing the question.
Alternatively, you can decline immediately by clicking:
[DECLINE_URL]

If you provide a review, the question's author will be notified and have the chance to revise 
the question and compose a response. After seeing the response, you have the option to submit 
a further review or recommend the question for acceptance in the benchmark.

Thank you for considering this request!

Best regards,
{self.get_inviter_name()}"""
        
        if is_external:
            template += """

Note: To track your review and allow you to see the author's replies, accepting the review 
request will create a user account for you on our website. You can optionally set a password 
after submitting your review to log back in and e.g. contribute a question to the benchmark yourself."""
        
        return template
