"""passbook OTP Forms"""

from django import forms
from django.contrib.admin.widgets import FilteredSelectMultiple
from django.core.validators import RegexValidator
from django.utils.safestring import mark_safe
from django.utils.translation import ugettext_lazy as _
from django_otp.models import Device

from passbook.factors.forms import GENERAL_FIELDS
from passbook.factors.otp.models import OTPFactor

OTP_CODE_VALIDATOR = RegexValidator(r'^[0-9a-z]{6,8}$',
                                    _('Only alpha-numeric characters are allowed.'))


class PictureWidget(forms.widgets.Widget):
    """Widget to render value as img-tag"""

    def render(self, name, value, attrs=None, renderer=None):
        return mark_safe(f'<img src="{value}" />') # nosec


class OTPVerifyForm(forms.Form):
    """Simple Form to verify OTP Code"""
    order = ['code']

    code = forms.CharField(label=_('Code'), validators=[OTP_CODE_VALIDATOR],
                           widget=forms.TextInput(attrs={
                               'autocomplete': 'off',
                               'placeholder': 'Code'
                               }))

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        # This is a little helper so the field is focused by default
        self.fields['code'].widget.attrs.update({
            'autofocus': 'autofocus',
            'autocomplete': 'off'
        })


class OTPSetupForm(forms.Form):
    """OTP Setup form"""
    title = _('Set up OTP')
    device: Device = None
    qr_code = forms.CharField(widget=PictureWidget, disabled=True, required=False,
                              label=_('Scan this Code with your OTP App.'))
    code = forms.CharField(label=_('Code'), validators=[OTP_CODE_VALIDATOR],
                           widget=forms.TextInput(attrs={'placeholder': _('One-Time Password')}))

    tokens = forms.MultipleChoiceField(disabled=True, required=False)

    def clean_code(self):
        """Check code with new otp device"""
        if self.device is not None:
            if not self.device.verify_token(int(self.cleaned_data.get('code'))):
                raise forms.ValidationError(_("OTP Code does not match"))
        return self.cleaned_data.get('code')

class OTPFactorForm(forms.ModelForm):
    """Form to edit OTPFactor instances"""

    class Meta:

        model = OTPFactor
        fields = GENERAL_FIELDS + ['enforced']
        widgets = {
            'name': forms.TextInput(),
            'order': forms.NumberInput(),
            'policies': FilteredSelectMultiple(_('policies'), False)
        }