Как я могу сделать собственный виджет так, чтобы form.is_valid () возвращал true, вместо множества данных <select>input? - PullRequest
0 голосов
/ 17 февраля 2012

models.py:

class Tag(models.Model):
    name = models.CharField(max_length=100)
    description = models.CharField(max_length=500, null=True, blank=True)
    created = models.DateTimeField(auto_now_add=True)
    modified = models.DateTimeField(auto_now_add=True)

class Post(models.Model):
    user = models.ForeignKey(User)
    tag = models.ManyToManyField(Tag)
    title = models.CharField(max_length=100)
    content = models.TextField()
    created = models.DateTimeField(default=datetime.datetime.now)
    modified = models.DateTimeField(default=datetime.datetime.now)

    def __unicode__(self):
        return '%s,%s' % (self.title,self.content)


class PostModelForm(forms.ModelForm):
    class Meta:
        model = Post


class PostModelFormNormalUser(forms.ModelForm):
    class Meta:
        model = Post
        widgets = { 'tag' : TextInput() }
        exclude = ('user', 'created', 'modified')

    def __init__(self, *args, **kwargs):
        super(PostModelFormNormalUser, self).__init__(*args, **kwargs)      
        self.fields['tag'].help_text = None

views.py:

    if request.method == 'POST':
        form = PostModelFormNormalUser(request.POST)
        print form
        print form.errors           
        tagstring = form.data['tag']
        splitedtag = tagstring.split()

        if form.is_valid():
            temp = form.save(commit=False)
            temp.user_id = user.id
            temp.save()

            l = len(splitedtag)         
            for i in range(l):
                obj = Tag(name=splitedtag[i])
                obj.save()
                post.tag_set.add(obj)

            post = Post.objects.get(id=temp.id)
            return HttpResponseRedirect('/viewpost/' + str(post.id))
    else:
        form = PostModelFormNormalUser()
        context = {'form':form}
        return render_to_response('addpost.html', context, context_instance=RequestContext(request))

Здесь form.is_valid() всегда ложно, потому что он получает тег как строку из формы. Но он ожидает, что список будет входом form.data ['tag']. Может кто-нибудь сказать мне, как я могу это исправить?

Как мне написать собственный виджет для решения этой проблемы?

1 Ответ

1 голос
/ 17 февраля 2012

Я не думаю, что вам нужен собственный виджет (вам все еще нужен TextInput), вам нужно пользовательское Поле . Для этого вы должны создать подкласс django.forms.Field. К сожалению, по этой теме недостаточно документации:

Если встроенные классы полей не соответствуют вашим потребностям, вы можете легко создавать собственные классы полей. Для этого просто создайте подкласс django.forms.Field. Его единственные требования - чтобы он реализовал метод clean () и чтобы его метод init () принимал основные аргументы, упомянутые выше (обязательно, label, initial, widget, help_text).

Я нашел это сообщение в блоге , которое более подробно охватывает как пользовательские виджеты, так и поля. Автор не согласен с документацией, которую я цитировал выше - ее стоит перечитать.

Для вашей конкретной ситуации вы бы сделали что-то вроде этого (не проверено):

class MyTagField(forms.Field):
    default_error_messages = {
        'some_error': _(u'This is a message re: the somr_error!'),
    }

    def to_python(self, value):
        # put code here to coerce 'value' (raw data from your TextInput)
        # into the form your code will want (a list of Tag objects, perhaps)

    def validate(self, value):
        if <not valid for some reason>:
            raise ValidationError(self.error_messages['some_error'])

Тогда в вашей ModelForm:

class PostModelFormNormalUser(forms.ModelForm):
    tag = MyTagField()

    class Meta:
        model = Post
        exclude = ('user', 'created', 'modified')

    def __init__(self, *args, **kwargs):
        super(PostModelFormNormalUser, self).__init__(*args, **kwargs)      
        self.fields['tag'].help_text = None
...