в Django я по прежнему новичок, но что-то стараюсь понять и запомнить.
стояла задача расширить профиль пользователя дополнительными полями, статей на эту тему пресс и малелькая тележка, но не получалось, то у кого-то на сайте парсер html сильно злой и код надо было переписывать, у кого-то были ошибки в примерах и т.п.
но как-то оно получилось, попробую собрать все мысли в кучу в эту статью.
Многие знают о django-registration и вот от автора этого замечательного приложения существует django-profiles котором я и воспользовался.
устанавливаем django-profiles
[cc lang=”bash”]easy_install django-profiles[/cc]
в models.py создадим модель профиля, которая наследуется от штатной User и добавим в ней нужное нам поле
[cc lang=”python”]
from django.contrib.auth.models import User
…
…
class UserProfile(models.Model):
# поле для связки со встроенной моделью пользователя Django
user = models.ForeignKey(User, unique=True)
# наше новое поле
about = models.TextField(blank=True)[/cc]
Далее в settings.py надо указать кто будет новым AUTH_PROFILE_MODULE
[cc lang=”python”]AUTH_PROFILE_MODULE = ‘list.UserProfile'[/cc]
где list это ваше приложение в котором расположена модель расширяющая пользователя
в принципе на этом можно закончить, ибо обращаться теперь можно к
[cc lang=”bash”]{{ user.get_profile.about }}[/cc]
только нужно обратить внимание что надо пользоваться не context а RequestContext
и указать это в
TEMPLATE_CONTEXT_PROCESSORS файла settings.py
[cc lang=”python”]TEMPLATE_CONTEXT_PROCESSORS = (
…
‘django.contrib.auth.context_processors.auth’,
…
) [/cc]
но мы идём дальше
Теперь нам надо подключить django-profiles для просмотра\редактирования профиля
в файле urls.py
подключаем
[cc lang=”python”] …
(r’^profiles/’, include(‘profiles.urls’)),
…[/cc]
в settings.py также подключим приложение
[cc lang=”python”]INSTALLED_APPS = (
…
‘profiles’,
)
[/cc]
теперь создадим таблицы в базе
[cc lang=”python”]python manage.py syncdb[/cc]
приложение profiles использует вот такие шаблоны
[cc lang=”bash”]
profiles/create_profile.html
profiles/edit_profile.html
profiles/profile_detail.html
profiles/profile_list.html[/cc]
по названию понятно что к чему
для начала отредактируем edit_profile.html
[cc lang=”html”]{% extends “base.html” %}
{% block title %}Edit Profile{% endblock %}
{% block content %}
Edit contact info for {{ user }}
{% endblock content %}[/cc]
сохраняем, теперь можно обратиться к /profiles/edit/ то должно открыться форма редактирования профиля.
бывают ситуации, что в профиле есть какие-то поля, что запрещено редактировать пользователю, реализуется это довольно просто
создаём класс формы, наследуемой от общей формы, в котором перечисляем, то что нам исключить от вывода пользователю
forms.py
[cc lang=”python”]from django.db import models
from django.forms import ModelForm
from PROJ.apps.APP.models import UserProfile
………
class ProfileForm(forms.ModelForm):
class Meta:
model = UserProfile
exclude = (‘user’,’last_name’,)
[/cc]
from PROJ.apps.APP.models import UserProfile
проект и приложение где расположена расширенная модель
теперь в urls.py надо указать, что страницу редактирования профиля надо открывать с нужной нам формой
[cc lang=”python”]from PROJ.APP.forms import ProfileForm
……..
(‘^profiles/edit’, ‘profiles.views.edit_profile’, {‘form_class’: ProfileForm,}),
(r’^profiles/’, include(‘profiles.urls’)),
…….[/cc]
Должно получиться примерно так.. проверяем на странице редактирования профиля (/profiles/edit/).
Теперь сделаем страницу отображения профиля profile/profile_detail.html
вывести можно что угодно из профиля вот так
[cc lang=”python”]
Address 2:
{{ profile.address2 }}
[/cc]
где address2 это дополнительное поле
список пользователей с профилями можно вывести вот так
profiles/profile_list.html
[cc lang=”python”]
{% extends “base.html” %}
{% block content %}
Список пользователей
{% for p in object_list %}
{{ p.user }}
{% endfor %}
{% endblock %}
[/cc]
Теперь задача такая, сейчас профиль и пользователь живут почти своей жизнью, при регистрации пользователя профиль сам не создаётся это мы сейчас и решим сигналами.
в models.py добавим
[cc lang=”python”]
from django.db.models import signals
from bucket.signals import create_profile
# When model instance is saved, trigger creation of corresponding profile
signals.post_save.connect(create_profile, sender=User)[/cc]
создадим файл, если он не создан signals.py:
[cc lang=”python”]def create_profile(sender, instance, signal, created, **kwargs):
“””When user is created also create a matching profile.”””
from PROJ.apps.APP.models import UserProfile
if created:
UserProfile(user = instance).save()
# Do additional stuff here if needed, e.g.
# create other required related records
[/cc]
вот и всё, при регистрации нового пользователя автоматически будет создаваться к нему профиль, если у вас уже сейчас много пользователей, то создать для каждого профиль можно вот так
[cc lang=”python”]$ python manage.py shell
from django.contrib.auth.models import User
from bucket.models import Profile
users = User.objects.all()
for u in users:
try:
p = u.get_profile()
except p.DoesNotExist:
Profile(user = u).save()[/cc]
ещё коснусь вопроса редактирования своего email в профиль, пока ещё не совсем разобрался с этим.
пока есть только пример, делается это анналогично через forms.py
[cc lang=”python”]class ProfileForm(ModelForm):
def __init__(self, *args, **kwargs):
super(ProfileForm, self).__init__(*args, **kwargs)
try:
self.fields[’email’].initial = self.instance.user.email
# self.fields[‘first_name’].initial = self.instance.user.first_name
# self.fields[‘last_name’].initial = self.instance.user.last_name
except User.DoesNotExist:
pass
email = forms.EmailField(label=”Primary email”,help_text=”)
class Meta:
model = Profile
exclude = (‘user’,)
def save(self, *args, **kwargs):
“””
Update the primary email address on the related User object as well.
“””
u = self.instance.user
u.email = self.cleaned_data[’email’]
u.save()
profile = super(ProfileForm, self).save(*args,**kwargs)
return profile[/cc]
надеюсь вернусь к этому вопросу