Форма виджета для ввода текста с внешней ссылкой хотел! - PullRequest
0 голосов
/ 03 июня 2009

Я сохраняю поле imdb_id для моделей Movie в моей базе данных:

class Movie(models.Model):
    imdb_id = models.IntegerField('imdb ID', blank=True, null=True, unique=True)

    def _get_imdb_url(self):
        return self.imdb_id and 'http://www.imdb.com/title/tt%s/' % str(self.imdb_id).zfill(7) or ''

    def _set_imdb_url(self, imdb_url):
        self.imdb_id = int( re.compile(r'[^\d]').sub('', imdb_url))

    imdb_url = property(_get_imdb_url, _set_imdb_url)

И я хочу сделать специальный виджет для отображения внешней ссылки на imdb.com в форме администратора возле ввода текста для поля «imdb_id». Я думаю, что это может быть глобальный виджет для любого поля формы с внешней ссылкой, созданный с помощью специальной маски (в моем случае, эта маска 'http://www.imdb.com/title/tt%s/'). Я знаю, как написать виджет, но я не знаю, как нажать моя маска, определенная в моей модели Movie, для этого виджета. Я не хочу нарушать принцип DRY и определять эту маску в двух разных местах. А также она будет хорошим инструментом для той же цели с другими внешними ссылками в будущем.

Что вы думаете об этом виджете? Как это возможно реализовать? Может быть, кто-то написал это до меня?

Спасибо!

1 Ответ

0 голосов
/ 04 июня 2009

Итак, я решил не ждать помощи и написал этот виджет :

import re
from django import forms
class LinkFieldWidget(Widget):
    '''
    A TextField widget with previewing link, generated from field's value with special url mask
    '''
    def __init__(self, text, url='%s', *args, **kwargs):
        self.url = url
        self.text = text
        if not re.search('%', self.url):
            raise forms.ValidationError, u'Invalid URL mask'
        super(LinkFieldWidget, self).__init__(*args, **kwargs)

    def render(self, name, value, attrs=None):
        output = []
        output.append('''<input type="text" name="%s" value="%s" onkeyup="$('a#%s-link').attr('href', '%s'.replace('%s', this.value)).css({'display': (this.value ? 'inline' : 'none')})" />''' % (name, value or '', name, self.url, '%s'))
        if self.url and self.text:
            output.append('<span><a href="%s" id="%s-link" target="blank" style="display: %s;">%s</a></span>' % (value and self.url % value or '#', name, value and 'inline' or 'none', self.text))

        return mark_safe(u' '.join(output))

Пример использования его в админке:

from utils.widgets import LinkFieldWidget
class MovieAdminForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(MovieAdminForm, self).__init__(*args, **kwargs)
        self.fields['imdb_id'].widget = LinkFieldWidget(text='imdb', url=self.instance.imdb_link)

class MovieAdmin(admin.ModelAdmin):
    form = MovieAdminForm

Модель:

class Movie(models.Model):
    .....    
    imdb_link = 'http://www.imdb.com/title/tt%s/'
    def _get_imdb_url(self):
        return self.imdb_id and self.imdb_link % str(self.imdb_id).zfill(7) or ''
    def _set_imdb_url(self, imdb_url):
        self.imdb_id = int( re.compile(r'[^\d]').sub('', imdb_url))
    imdb_url = property(_get_imdb_url, _set_imdb_url)
...