Django-ProfilesとDjango-Registrationを使用したさまざまなユーザープロファイル
-
04-10-2019 - |
質問
私のmodels.py:
USER_TYPES = (
('D', 'Demo' ),
('F', 'Free' ),
('P', 'Premium'),
)
class BaseProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
user_type = models.CharField(max_length=1, blank=True, choices=USER_TYPES)
class DemoProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
demo = models.CharField(max_length=10, blank=True)
...
class FreeProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
free = models.CharField(max_length=10, blank=True)
...
class PremiumProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
premium = models.CharField(max_length=10, blank=True)
...
class ProxyProfile(BaseProfile):
class Meta:
proxy = True
def get_profile(self):
if self.user_type == 'D':
return DemoProfile._default_manager.get(user__id__exact=self.user_id)
elif self.user_type == 'F':
return FreeProfile._default_manager.get(user__id__exact=self.user_id)
else:
return PremiumProfile._default_manager.get(user__id__exact=self.user_id)
baseprofileを使用して、user_idを特定のuser_typeにマッピングします。以下に示すように、ProxyProfileをProxyとしてModelformに依存してuser_typeをロードするプロキシとして使用したかった
私のforms.pyの内容:
class ProfileForm(ModelForm):
...
class Meta:
model = ProxyProfile
exclude = ('user','user_type')
...
プロファイルは、urls.pyの次のコードを使用して、django-profilesに提供されます。
urlpatterns += patterns('',
url(r'^profiles/edit/', edit_profile,
{'form_class': ProfileForm},
name='profiles_edit_profile'),
(r'^profiles/',include('profiles.urls')),
)
また、settings.pyに設定しました:
AUTH_PROFILE_MODULE = 'main.ProxyProfile'
ユーザー登録中、すべてのDBデータが正しく入力されます(すべてが問題ないようです)。 django-registrationに渡されたフォームを使用して登録します。
urlpatterns += patterns('',
url(r'^register/$', register,
{'form_class': UserRegistrationForm},
name='registration.views.register'),
(r'', include('registration.urls')),
)
forms.pyから:
class UserRegistrationForm(RegistrationFormUniqueEmail, RegistrationFormTermsOfService):
utype = forms.ChoiceField(choices=USER_CHOICES)
def save(self, profile_callback=None):
new_user = RegistrationProfile.objects.create_inactive_user(username=self.cleaned_data['username'],
password.self.cleaned_data['password1'],
email=self.cleaned_data['email'],
)
new_base_profile = BaseProfile(user=new_user, user_type=self.cleaned_data['utype'])
if self.cleaned_data['utype'] == "D":
new_profile = DemoProfile(user=new_user)
if self.cleaned_data['utype'] == "F":
new_profile = FreeProfile(user=new_user)
if self.cleaned_data['utype'] == "P":
new_profile = PremiumProfile(user=new_user)
new_profile.save()
new_base_profile.save()
return new_user
登録フェーズは正常に機能します。
プロフィール編集/詳細ページに問題があります。プロキシプロファイルモデルでフィルタリングされ、プロファイルのフォームモデルとして使用される私のプロファイルはレンダリングされていません(プロファイル特定のフィールドがHTMLページにレンダリングされていないことがわかりません)多分これを行うための別の方法(Django Way :)ユーザーモデルに関連するuser_typeフィールドに応じて、プロファイルモデルをレンダリングします。
前もって感謝します :)
解決
わかりました、ついに私はこれをどのように行うことができるか考えました:)
私のmodels.pyで:
class BaseManager(models.Manager):
def get(self, **kwargs):
self.u = kwargs['user__id__exact']
self.bt = BaseProfile.manager.get(user__id__exact=self.u)
if self.bt.user_type == 'F':
return FreeProfile.objects.get(pk=self.u)
elif self.bt.user_type == 'I':
return PremiumProfile.objects.get(pk=self.u)
else:
return None
class BaseProfile(models.Model):
objects = BaseManager()
manager = UserManager()
user = models.OneToOneField(User, primary_key=True)
user_type = models.CharField(max_length=1, blank=True, choices=USER_TYPES)
class FreeProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
free = models.CharField(max_length=10, blank=True)
...
class PremiumProfile(models.Model):
user = models.OneToOneField(User, primary_key=True)
premium = models.CharField(max_length=10, blank=True)
...
カスタムマネージャーで-BaseManagerでは、get_profileが使用するget()メソッドを上書きすることにより、プロファイルオブジェクトを返します。 self.btを割り当てるときにカスタムマネージャーの再帰的な呼び出しを防ぐために、単に「マネージャー」という名前のUsermanagerを使用する必要があります
わかりました、これは私が望むものをAChiveするための半分の方法です。これで、Django-Profilesアプリを使用してユーザーに添付されているさまざまなプロファイルを表示できます。
次に、ModElformを使用して、ユーザープロファイルの編集フォームを準備したいと思います。ユーザーはさまざまなプロファイルを持つことができるので、このスニペットで提示されたマジックトリックを適用しました。 http://djangosnippets.org/snippets/2081/
そして今、私の形で私は持っています:
class FreeForm(forms.ModelForm):
class Meta:
model = FreeProfile
class PremiumForm(forms.ModelForm):
class Meta:
model = PremiumProfile
次に、各プロファイルの単純なモデルフォームは、プロファイルで組み立てられます。
class ProfileForm(ModelForm):
def __init__(self, *args, **kwargs):
self.user = kwargs['instance'].user
profile_kwargs = kwargs.copy()
profile_kwargs['instance'] = self.user
self.bt = BaseProfile.manager.get(user__id__exact=self.user.id)
if self.bt.user_type == 'F':
self.profile_fields = FreeForm(*args, **profile_kwargs)
elif self.bt.user_type == 'P':
self.profile_fields = PremiumForm(*args, **profile_kwargs)
super(ProfileForm, self).__init__(*args, **kwargs)
self.fields.update(self.profile_fields.fields)
self.initial.update(self.profile_fields.initial)
class Meta:
model = BaseProfile
def save(self):
...
inestings.py:
AUTH_PROFILE_MODULE = 'main.BaseProfile'
そして、それは魅力のように機能しますが、それがDjangoプロファイルを使用して複数の異なるプロファイルのサポートを達成するためのDjangoの方法なのだろうか?プロファイルの詳細をレンダリングするか、フォームを編集する前に、get()をさらに数回使用する必要があることを心配しています。
しかし、これを成し遂げるためにダジャンゴとの苦労の4日間の後、私は今夜よく眠ることができます:)
乾杯