Подклассы django-registration 1.0 формы для пользовательских моделей django 1.5

django-registration 1.0 теперь поддерживает пользовательские модели django 1.5. Документация регистрации django содержит только следующий раздел:

Im, использующий Django 1.5 и пользовательскую модель пользователя; как это сделать?

Несмотря на то, что два встроенных django-registration оба предполагают модель Djangos по умолчанию, базовую классы просмотра преднамеренно не соответствуют пользовательской модели. Просто подкласс их и реализовать логику для пользовательской модели пользователя.

Я не уверен, какие представления мне нужны для подкласса и что им нужно делать. Я также заметил, что ProfileManager в django-registration по-прежнему принимает отдельное поле имени пользователя.

В моем конкретном случае я удалил поле "имя пользователя", добавил "display_name" и сделал "email" поле идентификации:

class MyUser(AbstractBaseUser, PermissionsMixin):
    email = models.EmailField(
        verbose_name="Email Address",
        max_length=384,
        unique=True,
        db_index=True,)
    display_name = models.CharField(max_length=128, blank=True)
    date_joined = models.DateTimeField(default=timezone.now)
    is_active = models.BooleanField(default=True)
    is_admin = models.BooleanField(default=False)

    objects = MyUserManager()

    USERNAME_FIELD = 'email'

    def get_full_name(self):
        return self.email

    def get_short_name(self):
        return self.email

    def __unicode__(self):
        return self.email

    def has_perm(self, perm, obj=None):
        return True

    def has_module_perms(self, app_label):
        return True

    @property
    def is_staff(self):
        return self.is_admin

Без подкласса каких-либо классов django-регистрации рендеринг по умолчанию регистрационной формы потянет поля из User вместо MyUser.

Я видел следующее SO-приложение django-registration app и пользовательскую модель Django 1.5, но это не помогло.

Обновление

Я заметил, что RegistrationForm жестко закодирован с полем "имя пользователя". В FAQ часто упоминается подклассу бэкэнда, поэтому я не уверен, что это за намерение. Должен ли я также подклассифицировать форму?

Ответ 1

Некоторые части наиболее определенно не совместимы с Django 1.5: https://bitbucket.org/ubernostrum/django-registration/src/8f242e35ef7c004e035e54b4bb093c32bf77c29f/registration/forms.py?at=default#cl-48

class RegistrationForm(forms.Form):
    # ...

    def clean_username(self):
        # ...
        # The line below needs fixing
        existing = User.objects.filter(username__iexact=self.cleaned_data['username'])
        if existing.exists():
            raise forms.ValidationError(_("A user with that username already exists."))
        else:
            return self.cleaned_data['username']

Поэтому, если эти методы не изменены и/или не подклассифицируют их, он пока не работает.

В вашем конкретном случае эта форма регистрации должна делать трюк:

from registration import forms as registration_forms
from django.contrib import auth

class RegistrationForm(registration_forms.RegistrationForm):
    def clean_username(self):
        '''
        Validate that the username is alphanumeric and is not already
        in use.
        '''
        User = auth.get_user_model()
        existing = User.objects.filter(display_name__iexact=self.cleaned_data['username'])
        if existing.exists():
            raise forms.ValidationError(_("A user with that name already exists."))
        else:
            return self.cleaned_data['username']

В дополнение к настраиваемому свойству вашей модели:

class MyUser(AbstractBaseUser, PermissionsMixin):
    # ...

    def get_username(self):
        return self.display_name

    def set_username(self, username):
        self.display_name = username

    def del_username(self):
        del self.display_name

    username = property(get_username, set_username, del_username)