質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.51%

  • Python 3.x

    9829questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

  • Django

    1610questions

    DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Django 会員登録フォームに2種類のモデルのデータを表示したい

受付中

回答 0

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 122

topy

score 2

前提・実現したいこと

ユーザー登録画面(/signup)に、下記の情報を表示させたいです。

・Userモデル定義したemail(unique=True)、password
・UserモデルとOneToOneでリレーションしている「Customerモデル」の情報(nickname、sex、birthday、profileImg)

なお、アプリはUserとCustomerで分けて作成しています。

現状は下記のエラーが出ている状況です。

type object 'Customer' has no attribute 'USERNAME_FIELD'


過去にhttps://teratail.com/questions/101936の投稿があり、直接リレーション先のオブジェクトを扱えるわけではない、とのことでしたが、どこを修正すれば/signupに全てのカラムを表示できるでしょうか。

forms.py、template、models.py、views.pyは下記のように記載している状況です。

from django import forms
from django.core.mail import send_mail
from django.conf import settings
from .models import Customer
from django.contrib.auth import get_user_model
from django.contrib.auth.forms import AuthenticationForm, UserCreationForm

User = get_user_model()

class UserCreateForm(UserCreationForm):
    """ユーザー登録用フォーム"""

    class Meta:
        model = Customer
        fields = '__all__'

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        for field in self.fields.values():
            field.widget.attrs['class'] = 'form-control'
<form action="" method="POST">
    {% csrf_token %}
    <div class="formStyle__itemBox">
        {{ form.user.email.label_tag }}
        {{ form.user.email }}
        {{ form.user.email.help_text }}
        {{ form.user.email.errors }}
    </div>
    <div class="formStyle__itemBox">
        {{ form.user.password.label_tag }}
        {{ form.user.password }}
        {{ form.user.password.help_text }}
        {{ form.user.password.errors }}
    </div>
    <div class="formStyle__itemBox">
        {{ form.nick_name.label_tag }}
        {{ form.nick_name }}
        {{ form.nick_name.help_text }}
        {{ form.nick_name.errors }}
    </div>
    <div class="formStyle__itemBox">
        {{ form.sex.label_tag }}
        {{ form.sex }}
        {{ form.sex.help_text }}
        {{ form.sex.errors }}
    </div>
    <div class="formStyle__itemBox">
        {{ form.birthday.label_tag }}
        {{ form.birthday }}
        {{ form.birthday.help_text }}
        {{ form.birthday.errors }}
    </div>
    <div class="formStyle__itemBox">
        {{ form.profile_img.label_tag }}
        {{ form.profile_img }}
        {{ form.profile_img.help_text }}
        {{ form.profile_img.errors }}
    </div>
    <button type="submit" class="btn btn-primary btn-lg">送信</button>
</form>
from django.db import models
from django.contrib.auth import get_user_model
from django.utils.translation import ugettext_lazy as _

User = get_user_model()

class Customer(models.Model):

    class Meta:
        verbose_name = verbose_name_plural = '00. カスタマー'

    user = models.OneToOneField(User, on_delete=models.CASCADE, related_name='customer')

    SEX = (
        ('man', '男性'),
        ('woman', '女性'),
        ('transgender', 'その他'),
    )

    nick_name = models.CharField(_('ニックネーム'), max_length=64)
    sex = models.CharField(_('性別'), max_length=12, choices=SEX)
    birthday = models.DateField(_('生年月日'), null=True)
    profile_img = models.ImageField(_('プロフィール画像'), upload_to='uploads/', blank=True)

    def __str__(self):
        return self.user.email
from django.db import models
from django.core.mail import send_mail
from django.contrib.auth import get_user_model
from django.contrib.auth.models import PermissionsMixin
from django.contrib.auth.models import AbstractBaseUser, BaseUserManager
from django.utils import timezone
from django.utils.translation import ugettext_lazy as _


class UserManager(BaseUserManager):

    use_in_migrations = True

    def _create_user(self, email, password, **extra_fields):
        if not email:
            raise ValueError('メールアドレスが入力されていません')
        email = self.normalize_email(email)

        user = self.model(email=email, **extra_fields)
        user.set_password(password)
        user.save(using=self._db)
        return user

    def create_user(self, email, password=None, **extra_fields):
        extra_fields.setdefault('is_staff', False)
        extra_fields.setdefault('is_superuser', False)
        return self._create_user(email, password, **extra_fields)

    def create_superuser(self, email, password, **extra_fields):
        extra_fields.setdefault('is_staff', True)
        extra_fields.setdefault('is_superuser', True)

        if extra_fields.get('is_staff') is not True:
            raise ValueError('スーパーユーザーはis_staff=Trueが必須です')
        if extra_fields.get('is_superuser') is not True:
            raise ValueError('スーパーユーザーはis_superuser=Trueが必須です')

        return self._create_user(email, password, **extra_fields)


class User(AbstractBaseUser, PermissionsMixin):
    class Meta:
        verbose_name = verbose_name_plural = '00. ユーザー'

    email = models.EmailField(_('メールアドレス'), max_length=255, unique=True, null=True)
    created = models.DateTimeField(_('作成日時'), auto_now_add=True)
    updated = models.DateTimeField(_('更新日時'), auto_now=True)

    is_staff = models.BooleanField(_('staff status'), default=False)
    is_active = models.BooleanField(_('有効フラグ'), default=True)

    date_joined = models.DateTimeField(_('date joined'), default=timezone.now)

    objects = UserManager()
    USERNAME_FIELD = 'email'

    def __str__(self):
        return self.email

    def email_user(self, subject, message, from_email=None, **kwargs):
        """Send an email to this user."""
        send_mail(subject, message, from_email, [self.email], **kwargs)

    @property
    def username(self):
        return self.email

    @property
    def is_owner(self):
        return hasattr(self, 'owner')
# 会員登録
class SignUpCreateView(generic.CreateView):
    """ユーザー仮登録"""
    template_name = 'customer/signup.html'
    form_class = UserCreateForm
    model = Customer

    def get_context_data(self, **kwargs):
        context = super(SignUpCreateView, self).get_context_data(**kwargs)

        context['user'] = User.objects.all()

        return context

    def form_valid(self, form):
        """仮登録と本登録用メールの発行."""
        # 仮登録と本登録の切り替えは、is_active属性を使うと簡単です。
        # 退会処理も、is_activeをFalseにするだけにしておくと捗ります。
        user = form.save(commit=False)
        user.is_active = False
        user.save()

        # アクティベーションURLの送付
        current_site = get_current_site(self.request)
        domain = current_site.domain
        context = {
            'protocol': self.request.scheme,
            'domain': domain,
            'token': dumps(user.pk),
            'user': user,
        }

        subject_template = get_template('customer/mail_template/create/subject.txt')
        subject = subject_template.render(context)

        message_template = get_template('customer/mail_template/create/message.txt')
        message = message_template.render(context)

        user.email_user(subject, message)
        return redirect('customer:signup_done')


class SignUpDoneTemplateView(generic.TemplateView):
    """ユーザー仮登録"""
    template_name = 'customer/signup_done.html'


class SignUpCompleteTemplateView(generic.TemplateView):
    """本登録用に発行したメール内のURLアクセス後のユーザー本登録"""
    template_name = 'customer/signup_complete.html'
    timeout_seconds = getattr(settings, 'ACTIVATION_TIMEOUT_SECONDS', 60*60*24)  # デフォルトでは1日以内

    def get(self, request, **kwargs):
        """tokenが正しければ本登録."""
        token = kwargs.get('token')
        try:
            user_pk = loads(token, max_age=self.timeout_seconds)

        # 期限切れ
        except SignatureExpired:
            return HttpResponseBadRequest()

        # tokenが間違っている
        except BadSignature:
            return HttpResponseBadRequest()

        # tokenは問題なし
        else:
            try:
                user = User.objects.get(pk=user_pk)
                customer = Customer()
                customer.user = User.objects.get(pk=user_pk)
                customer.save()
            except User.DoesNotExist:
                return HttpResponseBadRequest()
            else:
                if not user.is_active:
                    # 問題なければ本登録とする
                    user.is_active = True
                    user.save()
                    return super().get(request, **kwargs)

        return HttpResponseBadRequest()

補足情報(FW/ツールのバージョンなど)

Python:3.7
Django:2.1.2

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

まだ回答がついていません

同じタグがついた質問を見る

  • Python 3.x

    9829questions

    Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

  • Django

    1610questions

    DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。