Administrator django ustawia pole jako tylko do odczytu podczas modyfikowania obj, ale wymagane przy dodawaniu nowego obiektu

91

W panelu administracyjnym chciałbym wyłączyć pole podczas modyfikowania obiektu, ale wymagam tego przy dodawaniu nowego obiektu.

Jaki jest sposób na to django?

frnhr
źródło

Odpowiedzi:

178

Możesz zmienić get_readonly_fieldsmetodę administratora :

class MyModelAdmin(admin.ModelAdmin):

    def get_readonly_fields(self, request, obj=None):
        if obj: # editing an existing object
            return self.readonly_fields + ('field1', 'field2')
        return self.readonly_fields
Bernhard Vallant
źródło
21
Drobne / główne zastrzeżenie: to nie działa w przypadku wbudowanych. Dynamiczny przycisk „dodaj kolejny X” pokazuje pole tylko do odczytu jako „(Brak)”, a nie pole formularza, jak można się spodziewać.
Cerin
17

Jeśli chcesz ustawić wszystkie pola jako tylko do odczytu tylko w widoku zmian, nadpisz get_readonly_fields administratora:

def get_readonly_fields(self, request, obj=None):
    if obj: # editing an existing object
        # All model fields as read_only
        return self.readonly_fields + tuple([item.name for item in obj._meta.fields])
    return self.readonly_fields

A jeśli chcesz ukryć przyciski zapisywania w widoku zmiany :

  1. Zmień widok

    def change_view(self, request, object_id, form_url='', extra_context=None):
        ''' customize edit form '''
        extra_context = extra_context or {}
        extra_context['show_save_and_continue'] = False
        extra_context['show_save'] = False
        extra_context['show_save_and_add_another'] = False # this not works if has_add_permision is True
        return super(TransferAdmin, self).change_view(request, object_id, extra_context=extra_context)
    
  2. Zmień uprawnienia, jeśli użytkownik próbuje edytować:

    def has_add_permission(self, request, obj=None):
       # Not too much elegant but works to hide show_save_and_add_another button
        if '/change/' in str(request):
            return False
        return True
    

    To rozwiązanie zostało przetestowane w Django 1.11

DVicenteR
źródło
Idealny. Właśnie tego potrzebowałem!
wogsland,
3

FYI: na wypadek, gdyby ktoś inny napotkał te same dwa problemy, które napotkałem:

  1. Nadal powinieneś zadeklarować wszystkie trwale readonly_fields w treści klasy, ponieważ atrybut klasy readonly_fields będzie dostępny z walidacji (patrz django.contrib.admin.validation: validate_base (), line.213 appx)

  2. To nie zadziała z Inlines, ponieważ obiekt przekazany do get_readonly_fields () jest obiektem nadrzędnym (mam dwa dość hakerskie i nisko zabezpieczone rozwiązania używające css lub js)

Tim Diggins
źródło
2
2. punkt - to z powodu błędu w panelu administracyjnym: # 15602 Wygląda na to, że nie zostanie to wkrótce naprawione (ostatnia aktywność 2 lata temu), więc wygląda na to, że zostaliśmy pozostawieni rozwiązaniom CSS / JS.
frnhr
2

Odmiana oparta na poprzedniej doskonałej sugestii Bernharda Vallanta, która również zachowuje wszelkie możliwe dostosowania zapewniane przez klasę podstawową (jeśli istnieje):

class MyModelAdmin(BaseModelAdmin):

    def get_readonly_fields(self, request, obj=None):
        readonly_fields = super(MyModelAdmin, self).get_readonly_fields(request, obj)
        if obj: # editing an existing object
            return readonly_fields + ['field1', ..]
        return readonly_fields
Mario Orlandi
źródło
2

Sytuacja z formularzami wbudowanymi nadal nie została rozwiązana w Django 2.2.x, ale rozwiązanie Johna jest całkiem sprytne.

Kod nieco dostosowany do mojej sytuacji:

class NoteListInline(admin.TabularInline):
""" Notes list, readonly """
    model = Note
    verbose_name = _('Note')
    verbose_name_plural = _('Notes')
    extra = 0
    fields = ('note', 'created_at')
    readonly_fields = ('note', 'created_at')

    def has_add_permission(self, request, obj=None):
    """ Only add notes through AddInline """
    return False

class NoteAddInline(admin.StackedInline):
    """ Notes edit field """
    model = Note
    verbose_name = _('Note')
    verbose_name_plural = _('Notes')
    extra = 1
    fields = ('note',)
    can_delete = False

    def get_queryset(self, request):
        queryset = super().get_queryset(request)
        return queryset.none()  # no existing records will appear

@admin.register(MyModel)
class MyModelAdmin(admin.ModelAdmin):
    # ...
    inlines = (NoteListInline, NoteAddInline)
    # ...
jlapoutre
źródło
0

Możesz to zrobić, zastępując metodę formfield_for_foreignkey w ModelAdmin:

from django import forms
from django.contrib import admin

from yourproject.yourapp.models import YourModel

class YourModelAdmin(admin.ModelAdmin):

    class Meta:
        model = YourModel

    def formfield_for_foreignkey(self, db_field, request=None, **kwargs):
        # Name of your field here
        if db_field.name == 'add_only':
            if request:
                add_opts = (self._meta.app_label, self._meta.module_name)
                add = u'/admin/%s/%s/add/' % add_opts
                if request.META['PATH_INFO'] == add:
                    field = db_field.formfield(**kwargs)
                else:
                    kwargs['widget'] = forms.HiddenInput()
                    field = db_field.formfield(**kwargs)
            return field
        return admin.ModelAdmin(self, db_field, request, **kwargs)
David Miller
źródło
0

Mam podobny problem. Rozwiązałem to za pomocą „add_fieldsets” i „limited_fieldsets” w ModelAdmin.

from django.contrib import admin  
class MyAdmin(admin.ModelAdmin):
 declared_fieldsets = None
 restricted_fieldsets = (
    (None, {'fields': ('mod_obj1', 'mod_obj2')}),
    ( 'Text', {'fields': ('mod_obj3', 'mod_obj4',)}),
 )

 add_fieldsets = (
            (None, {
             'classes': ('wide',),
             'fields': ('add_obj1', 'add_obj2', )}),
             )

Zobacz np .: http://code.djangoproject.com/svn/django/trunk/django/contrib/auth/admin.py

Ale to nie chroni twojego modelu przed późniejszymi zmianami „add_objX”. Jeśli chcesz tego również, myślę, że musisz przejść przez funkcję „save” klasy Model i sprawdzić tam zmiany.

Zobacz: www.djangoproject.com/documentation/models/save_delete_hooks/

Greez, Nick

Nick Ma.
źródło