Как добавить ссылку со страницы администрирования Django одного объекта на страницу администрирования связанного объекта? - PullRequest
27 голосов
/ 29 марта 2012

Чтобы справиться с отсутствием вложенных строк в django-admin, я поместил особые случаи в два шаблона для создания ссылок между страницами смены администратора и встроенными администраторами двух моделей.

Мой вопрос таков: как создать ссылку со страницы смены администратора или встроенного администратора одной модели на страницу смены администратора или встроенного администратора связанной модели без каких-либо неприятных взломов в шаблоне?

Мне бы хотелось общее решение, которое я могу применить к странице смены администратора или встроенному администратору любой модели.


У меня есть одна модель, post (не еенастоящее имя), которое является одновременно встроенным на странице администрирования blog, а также имеет собственную страницу администратора.Причина, по которой он не может быть просто встроенным, состоит в том, что у него есть модели с внешними ключами, которые имеют смысл только при редактировании с ним, и это имеет смысл только при редактировании с помощью blog.

Для post Страница администратора, я изменил часть "fieldset.html" с:

{% if field.is_readonly %}
    <p>{{ field.contents }}</p>
{% else %}
    {{ field.field }}
{% endif %}

на

{% if field.is_readonly %}
    <p>{{ field.contents }}</p>
{% else %}
    {% ifequal field.field.name "blog" %}
        <p>{{ field.field.form.instance.blog_link|safe }}</p>
    {% else %}
        {{ field.field }}
    {% endifequal %}
{% endif %}

, чтобы создать ссылку на страницу администрирования blog, где blog_linkметод в модели:

def blog_link(self):
      return '<a href="%s">%s</a>' % (reverse("admin:myblog_blog_change",  
                                        args=(self.blog.id,)), escape(self.blog))

Я не смог найти id экземпляра blog за пределами field.field.form.instance.

На странице администрирования blogгде post является встроенным, я изменил часть "stacked.html" с:

<h3><b>{{ inline_admin_formset.opts.verbose_name|title }}:</b>&nbsp;
<span class="inline_label">{% if inline_admin_form.original %}
    {{ inline_admin_form.original }}
{% else %}#{{ forloop.counter }}{% endif %}</span>

до

<h3><b>{{ inline_admin_formset.opts.verbose_name|title }}:</b>&nbsp;
<span class="inline_label">{% if inline_admin_form.original %}
    {% ifequal inline_admin_formset.opts.verbose_name "post" %}
    <a href="/admin/myblog/post/{{ inline_admin_form.pk_field.field.value }}/">
            {{ inline_admin_form.original }}</a>
{% else %}{{ inline_admin_form.original }}{% endifequal %}
{% else %}#{{ forloop.counter }}{% endif %}</span>

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


Я уверен, что есть лучший, более общий способ добавления ссылок на формы администратора без повторениясебя;что это?

Ответы [ 7 ]

14 голосов
/ 03 апреля 2012

Использовать readonly_fields :

class MyInline(admin.TabularInline):
    model = MyModel
    readonly_fields = ['link']

    def link(self, obj):
        url = reverse(...)
        return mark_safe("<a href='%s'>edit</a>" % url)

    # the following is necessary if 'link' method is also used in list_display
    link.allow_tags = True
14 голосов
/ 28 июля 2015

Новое в Django 1.8: show_change_link для встроенного администратора .

Установите show_change_link в True (по умолчанию False) в вашей встроенной модели, так что встроенные объекты имеют ссылку на свою форму изменения (где они могут иметь свои собственные встроенные строки).

from django.contrib import admin

class PostInline(admin.StackedInline):
    model = Post
    show_change_link = True
    ...

class BlogAdmin(admin.ModelAdmin):
    inlines = [PostInline]
    ...

class ImageInline(admin.StackedInline):
    # Assume Image model has foreign key to Post
    model = Image
    show_change_link = True
    ...

class PostAdmin(admin.ModelAdmin):
    inlines = [ImageInline]
    ...

admin.site.register(Blog, BlogAdmin)
admin.site.register(Post, PostAdmin)
10 голосов
/ 08 ноября 2012

Я думаю, что решение agf довольно круто - ему много благодарностей. Но мне нужно было еще пару функций:

  • чтобы иметь возможность иметь несколько ссылок для одного администратора
  • для ссылки на модель в другом приложении

Решение:

def add_link_field(target_model = None, field = '', app='', field_name='link',
                   link_text=unicode):
    def add_link(cls):
        reverse_name = target_model or cls.model.__name__.lower()
        def link(self, instance):
            app_name = app or instance._meta.app_label
            reverse_path = "admin:%s_%s_change" % (app_name, reverse_name)
            link_obj = getattr(instance, field, None) or instance
            url = reverse(reverse_path, args = (link_obj.id,))
            return mark_safe("<a href='%s'>%s</a>" % (url, link_text(link_obj)))
        link.allow_tags = True
        link.short_description = reverse_name + ' link'
        setattr(cls, field_name, link)
        cls.readonly_fields = list(getattr(cls, 'readonly_fields', [])) + \
            [field_name]
        return cls
    return add_link

Использование:

# 'apple' is name of model to link to
# 'fruit_food' is field name in `instance`, so instance.fruit_food = Apple()
# 'link2' will be name of this field
@add_link_field('apple','fruit_food',field_name='link2')
# 'cheese' is name of model to link to
# 'milk_food' is field name in `instance`, so instance.milk_food = Cheese()
# 'milk' is the name of the app where Cheese lives
@add_link_field('cheese','milk_food', 'milk')
class FoodAdmin(admin.ModelAdmin):
    list_display = ("id", "...", 'link', 'link2')

Я сожалею, что пример настолько нелогичен, но я не хотел использовать мои данные.

10 голосов
/ 04 апреля 2012

Это мое текущее решение, основанное на том, что было предложено Панну (в его редакции) и Михаилом.

У меня есть пара представлений о смене администратора верхнего уровня, мне нужно связать с представлением о смене администратора верхнего уровня связанного объекта, и пара встроенных представлений о смене администратора, мне нужно связать с изменением администратора верхнего уровня вид на тот же объект. В связи с этим я хочу выделить метод связи, а не повторять его вариации для каждого представления изменений администратора.

Я использую декоратор класса для создания link вызываемого и добавляю его к readonly_fields.

def add_link_field(target_model = None, field = '', link_text = unicode):
    def add_link(cls):
        reverse_name = target_model or cls.model.__name__.lower()
        def link(self, instance):
            app_name = instance._meta.app_label
            reverse_path = "admin:%s_%s_change" % (app_name, reverse_name)
            link_obj = getattr(instance, field, None) or instance
            url = reverse(reverse_path, args = (link_obj.id,))
            return mark_safe("<a href='%s'>%s</a>" % (url, link_text(link_obj)))
        link.allow_tags = True
        link.short_description = reverse_name + ' link'
        cls.link = link
        cls.readonly_fields = list(getattr(cls, 'readonly_fields', [])) + ['link']
        return cls
    return add_link

Вы также можете передать настраиваемый вызываемый объект, если вам нужно каким-то образом получить текст ссылки, чем просто вызвать unicode для объекта, на который вы ссылаетесь.

Я использую это так:

# the first 'blog' is the name of the model who's change page you want to link to
# the second is the name of the field on the model you're linking from
# so here, Post.blog is a foreign key to a Blog object. 
@add_link_field('blog', 'blog')
class PostAdmin(admin.ModelAdmin):
    inlines = [SubPostInline, DefinitionInline]
    fieldsets = ((None, {'fields': (('link', 'enabled'),)}),)
    list_display = ('__unicode__', 'enabled', 'link')

# can call without arguments when you want to link to the model change page
# for the model of an inline model admin.
@add_link_field()
class PostInline(admin.StackedInline):
    model = Post
    fieldsets = ((None, {'fields': (('link', 'enabled'),)}),)
    extra = 0

Конечно, ничего из этого не потребовалось бы, если бы я мог вкладывать представления смены администратора для SubPost и Definition во встроенного администратора Post на странице смены администратора Blog без исправления Django.

3 голосов
/ 29 марта 2012

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

ИтакЯ использовал виджет привязки вместе с переопределением формы, чтобы получить ссылку на странице.

forms.py:

class AnchorWidget(forms.Widget):

    def _format_value(self,value):
        if self.is_localized:
            return formats.localize_input(value)
        return value

    def render(self, name, value, attrs=None):
        if not value:
            value = u''

        text = unicode("")
        if self.attrs.has_key('text'):
            text = self.attrs.pop('text')

        final_attrs = self.build_attrs(attrs,name=name)

        return mark_safe(u"<a %s>%s</a>" %(flatatt(final_attrs),unicode(text)))

class PostAdminForm(forms.ModelForm):
    .......

    def __init__(self,*args,**kwargs):
        super(PostAdminForm, self).__init__(*args, **kwargs)
        instance = kwargs.get('instance',None)
        if instance.blog:
            href = reverse("admin:appname_Blog_change",args=(instance.blog))  
            self.fields["link"] = forms.CharField(label="View Blog",required=False,widget=AnchorWidget(attrs={'text':'go to blog','href':href}))


 class BlogAdminForm(forms.ModelForm):
    .......
    link = forms..CharField(label="View Post",required=False,widget=AnchorWidget(attrs={'text':'go to post'}))

    def __init__(self,*args,**kwargs):
        super(BlogAdminForm, self).__init__(*args, **kwargs)
        instance = kwargs.get('instance',None)
        href = ""
        if instance:
            posts = Post.objects.filter(blog=instance.pk)
            for idx,post in enumerate(posts):
                href = reverse("admin:appname_Post_change",args=(post["id"]))  
                self.fields["link_%s" % idx] = forms..CharField(label=Post["name"],required=False,widget=AnchorWidget(attrs={'text':post["desc"],'href':href}))

теперь в вашем ModelAdminпереопределите атрибут form, и вы получите желаемый результат.Я предположил, что между этими таблицами есть отношение OneToOne. Если у вас есть один ко многим, то сторона BlogAdmin не будет работать.

обновление: Я внес некоторые изменения для динамического добавления ссылок, и это также решает проблему OneToMany с Blog до Post, надеюсь, это решит проблему.:)

После Pastebin: В вашем PostAdmin я заметил blog_link, это означает, что вы пытаетесь показать ссылку blog на changelist_view, в которой перечислены все сообщения.Если я прав, то вы должны добавить метод, чтобы показать ссылку на странице.

class PostAdmin(admin.ModelAdmin):
    model = Post
    inlines = [SubPostInline, DefinitionInline]
    list_display = ('__unicode__', 'enabled', 'blog_on_site')

    def blog_on_site(self, obj):
        href = reverse("admin:appname_Blog_change",args=(obj.blog))
        return mark_safe(u"<a href='%s'>%s</a>" %(href,obj.desc))
    blog_on_site.allow_tags = True
    blog_on_site.short_description = 'Blog'

Насколько отображает post ссылки на BlogAdmin changelist_view, вы можете сделать так же, каквыше.Мое раннее решение покажет вам ссылку на один уровень ниже на странице change_view, где вы можете редактировать каждый экземпляр.

Если вы хотите, чтобы на странице BlogAdmin отображались ссылки на post в change_view страницы, то вам нужно будет динамически включить каждую из них в fieldsets, переопределив метод get_form для class BlogAdmin и добавив ссылку динамически, в get_form установите self.fieldsets, но сначала не используйте кортеживместо fieldsets используйте список.

2 голосов
/ 04 сентября 2014

Основываясь на предложениях agfs и SummerBreeze, я улучшил декоратор, чтобы лучше обрабатывать Юникод и иметь возможность связывать поля с обратными внешними ключами (ManyRelatedManager с одним результатом).Также теперь вы можете добавить short_description в качестве заголовка списка:

from django.core.urlresolvers import reverse
from django.core.exceptions import MultipleObjectsReturned
from django.utils.safestring import mark_safe

def add_link_field(target_model=None, field='', app='', field_name='link',
                   link_text=unicode, short_description=None):
    """
    decorator that automatically links to a model instance in the admin;
    inspired by /4936292/dobavit-ssylku-stranitsy-administrirovaniya-django-odnogo-obekta-stranitsu-administrirovaniya-svyazannogo-obekta
    to-the-admin-page-o
    :param target_model: modelname.lower or model
    :param field: fieldname
    :param app: appname
    :param field_name: resulting field name
    :param link_text: callback to link text function
    :param short_description: list header
    :return:
    """
    def add_link(cls):
        reverse_name = target_model or cls.model.__name__.lower()

        def link(self, instance):
            app_name = app or instance._meta.app_label
            reverse_path = "admin:%s_%s_change" % (app_name, reverse_name)
            link_obj = getattr(instance, field, None) or instance

            # manyrelatedmanager with one result?
            if link_obj.__class__.__name__ == "RelatedManager":
                try:
                    link_obj = link_obj.get()
                except MultipleObjectsReturned:
                    return u"multiple, can't link"
                except link_obj.model.DoesNotExist:
                    return u""

            url = reverse(reverse_path, args = (link_obj.id,))
            return mark_safe(u"<a href='%s'>%s</a>" % (url, link_text(link_obj)))
        link.allow_tags = True
        link.short_description = short_description or (reverse_name + ' link')
        setattr(cls, field_name, link)
        cls.readonly_fields = list(getattr(cls, 'readonly_fields', [])) + \
            [field_name]
        return cls
    return add_link

Редактировать: обновлено из-за отсутствия ссылки.

0 голосов
/ 04 апреля 2012

Просматривая источник административных классов, поучительно: он показывает, что в контексте есть объект, доступный для представления администратора, называемый «оригинальный».

Вот аналогичная ситуация, когда мне понадобилась некоторая информациядобавлено в список изменений: Добавление данных в шаблоны администратора (в моем блоге).

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...