Django Image没有保存在数据库中

时间:2017-11-22 18:02:39

标签: python html django django-models django-database

我试图上传用户照片并在主页上显示。

问题1:当我从管理部分上传图片时,它会保存在数据库中,但它不会显示在' homepage.html'中。

问题2:当我尝试使用' profile.html'上传图片时模板图像甚至没有保存在数据库中。

我已尝试过stackoverflow的几个解决方案但未能解决我的问题。我的代码如下:

models.py

from django.db import models
from django.contrib.auth.models import User
from django.db.models.signals import post_save
from django.dispatch import receiver


class UserProfile(models.Model):
    user = models.OneToOneField(User, on_delete=models.CASCADE, related_name='profile')
    first_name= models.CharField(max_length=100, default='', blank=True)
    last_name = models.CharField(max_length=100, default='', blank=True)
    email = models.CharField(max_length=100, default='', blank=True)
    pro_pic = models.ImageField(upload_to='profilepic',blank=True)
    phone = models.CharField(max_length=20, blank=True, default='')
    city = models.CharField(max_length=100, default='', blank=True)
    country = models.CharField(max_length=100, default='', blank=True)
    job = models.CharField(max_length=20, blank=True, default='')
    organization = models.CharField(max_length=100, default='', blank=True)

    def __str__(self):
        return self.user.username

    @receiver(post_save, sender=User)
    def create_user_profile(sender, instance, created, **kwargs):
        if created:
            UserProfile.objects.create(user=instance)

    @receiver(post_save, sender=User)
    def save_user_profile(sender, instance, **kwargs):
        instance.profile.save()

forms.py

from django.contrib.auth.models import User
from django import forms
from .models import UserProfile


class UserForm(forms.ModelForm):
    class Meta:
        model = UserProfile
        fields = ['first_name', 'last_name', 'email']

class ProfileForm(forms.ModelForm):
    class Meta:
        model = UserProfile
        fields = ['pro_pic','phone', 'city', 'country', 'job', 'organization']

settings.py

MEDIA_URL = '/media/'
MEDIA_ROOT=os.path.join(BASE_DIR,'obs/media')

views.py

@login_required
@transaction.atomic
def update_profile(request):
    if request.method == 'POST':
        user_form = UserForm(request.POST, instance=request.user)
        profile_form = ProfileForm(request.POST, instance=request.user.profile)
        if user_form.is_valid() and profile_form.is_valid():
            user_form.save()
            profile_form.save()
            messages.success(request,'Your profile was successfully updated!')
            return redirect('obs:homepage')
        else:
            messages.error(request, 'Please correct the error below.')
    else:
        user_form = UserForm(instance=request.user)
        profile_form = ProfileForm(instance=request.user.profile)
    return render(request, 'obs/profile.html', {
        'user_form': user_form,
        'profile_form': profile_form
    })

profile.html

{% extends 'obs/base.html' %}


{% block body %}
{% load staticfiles %}
    <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css">
    <link href='https://fonts.googleapis.com/css?family=Satisfy' rel='stylesheet' type='text/css'>
    <link rel="stylesheet" type="text/css" href="{% static 'obs/profilecss.css' %}"/>
    <script src="https://ajax.googleapis.com/ajax/libs/jquery/1.12.0/jquery.min.js"></script>
    <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/js/bootstrap.min.js"></script>



<head>
    <title>Profile</title>
</head>

<div class="container">
    <div class="row">
        <div class="col-md-5 col-md-offset-3">
            <div class="panel panel-default">
                <div class="panel-heading">
                    <span class="glyzphicon glyphicon-user"></span> Update User Profile</div>

                <div class="panel-body">
                    {% if error_message %}
                        <p><strong>{{ error_message }}</strong></p>
                    {% endif %}
                 <form method="post" class="hello">
                  {% csrf_token %}

                 {% for field in user_form %}

                    <div class="form-group">
                        <label  class="col-sm-15 control-label">
                            {{ field.label_tag }}</label>
                        {{ field }}

                    </div>

                {% endfor %}

                {% for field in profile_form %}

                    <div class="form-group">
                        <label  class="col-sm-15 control-label">
                            {{ field.label_tag }}</label>
                        {{ field }}

                    </div>

                {% endfor %}
                  <center>
                  <div class="buttonHolder">
                      <button class="button tick" type="submit"></button>


                  </div>
                 </center>
                </form>
                </div>
            </div>
        </div>
    </div>
</div>

{% endblock %}

homepage.html

<div class="profile-userpic">
    <img src="{{ profile_form.pro_pic.url }}" class="img-responsive" alt="">
</div>

urls.py(项目)

from django.conf.urls import include, url
from django.contrib import admin
from django.conf import settings
from django.conf.urls.static import static



urlpatterns = [
    url(r'^admin/', admin.site.urls),
    url(r'^onlinebookshare/', include('registration.urls', namespace="accounts")),

]+ static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

查看主页:

@login_required
@transaction.atomic
def update_home(request):

    user_form = UserForm(instance=request.user)
    profile_form = ProfileForm(instance=request.user.profile)
    return render(request, 'obs/homepage.html', {
        'user_form': user_form,
        'profile_form': profile_form
    })

2 个答案:

答案 0 :(得分:1)

首先,您需要按照指定创建 MEDIA_ROOT 目录和 upload_to 目录。

MEDIA_ROOT=os.path.join(BASE_DIR,'obs/media')
pro_pic = models.ImageField(upload_to='profilepic',blank=True)

在您上传个人资料图片的表单标签中,您需要添加以下 enctype =&#39; multipart / form-data&#39; ,如

<form method="post" enctype="multipart/form-data" class="hello">.

在update_profile视图中,更改以下行:

profile_form = ProfileForm(request.POST, instance=request.user.profile) 

profile_form = ProfileForm(request.POST, instance=request.user.profile, request.FILES or None)

希望这会有所帮助。

答案 1 :(得分:0)

你做的事情有点奇怪。您正尝试通过表单访问已上传的文件,其URL存储在模型的字段中。问题的直接原因是表单字段不知道URL,它只有一个值。您需要通过底层表单实例访问它,您将其作为配置文件传递 - {{ profile_form.instance.pro_pic.url }}

但我不知道你为什么要这样做。无论如何,显示事物不是形式的责任。您应该直接使用配置文件,该配置文件已通过用户对象提供。

{{ user.profile.pro_pic.url }}

对于第二个问题,您需要将上传的文件传递给表单:

profile_form = ProfileForm(request.POST, request.FILES, instance=request.user.profile)