2017-10-17 4 views
0
내가 견적 (마스터) 헤더 정보와 합계를 포함하는 간단한 마스터/세부 모델이

QuotationDetail (세부 정보)를 업데이트 할 때 견적 항목의 경우이 자식 모델에는 PriceLevel FK 필드이 포함되어 각 개별 항목의 마크 업을 계산합니다. 나는 관리자가 한 번에 여러 항목에서이 PriceLevel을 변경할 수 있도록 장고 관리자 액션을 구현하기 위해 노력하고 있어요,하지만 난 오류가 발생합니다 :파이썬 장고 1.11 "ValueError를 불완전한 형식은"FK 필드를 포함

[myProjectPath]\lib\site-packages\django\contrib\admin\options.py", line 812, in get_action_choices 
    choice = (name, description % model_format_dict(self.opts)) 
ValueError: incomplete format 

이 (해당 모델) 내 models.py입니다 :

class NivelDePrecio(models.Model): # PriceLevel 
    # Fields 
    nombre = models.CharField(max_length=50) 
    slug = extension_fields.AutoSlugField(populate_from='nombre', blank=True) 
    valor = models.DecimalField(max_digits=7, decimal_places=4) 
    factor = models.DecimalField(max_digits=7, decimal_places=6, null=True, blank=True) 

    class Meta: 
     ordering = ('-valor',) 

    def __str__(self): 
     return self.nombre 

    def save(self, *args, **kwargs): 
     self.factor = 1/(1 - self.valor) 
     super(NivelDePrecio, self).save() 


class Cotizacion(models.Model): # Quotation (master) 
    # Fields 
    nombre = models.CharField(max_length=100) 
    slug = extension_fields.AutoSlugField(populate_from='nombre', blank=True, overwrite=True) 
    fecha_ida = models.DateField(default=date.today) 
    fecha_regreso = models.DateField(default=date.today) 

    # Relationship Fields 
    itinerario = models.ForeignKey(Itinerario, on_delete=CASCADE, verbose_name='itinerario') 
    nivel_de_precio = models.ForeignKey(NivelDePrecio, verbose_name='nivel de precio', 
             on_delete=models.PROTECT, null=True, blank=True) 

    class Meta: 
     ordering = ('itinerario__cliente__codigo', '-fecha_ida') 

    def __str__(self): 
     return str(self.nombre) 


class CotizacionDetalle(models.Model): # QuotationDetail (detail) 
    # Fields 
    descripcion = models.CharField(max_length=100, null=True, blank=True) 
    cantidad = models.DecimalField(max_digits=10, decimal_places=2) 
    costo = models.DecimalField(max_digits=10, decimal_places=2, default=0, editable=False) 
    monto = models.DecimalField(max_digits=10, decimal_places=2, default=0, editable=False) 
    markup = models.DecimalField(max_digits=6, decimal_places=4, default=0, editable=False) 
    utilidad = models.DecimalField(max_digits=6, decimal_places=4, default=0, editable=False) 
    total = models.DecimalField(max_digits=10, decimal_places=2, default=0, editable=False) 
    slug = extension_fields.AutoSlugField(populate_from='item', blank=True) 

    # Relationship Fields 
    cotizacion = models.ForeignKey(Cotizacion, on_delete=CASCADE, verbose_name='cotizacion', related_name='lineas') 
    item = models.ForeignKey(Item, verbose_name='item', related_name='item') 
    nivel_de_precio = models.ForeignKey(NivelDePrecio, verbose_name='nivel de precio', 
             on_delete=models.PROTECT, null=True, blank=True) 

    class Meta: 
     ordering = ('id',) 

    def save(self, *args, **kwargs): 
     self.descripcion = self.item.descripcion_venta 
     self.costo = self.item.costo 
     self.monto = self.cantidad * self.costo 
     if self.nivel_de_precio is None: 
      self.nivel_de_precio = self.cotizacion.nivel_de_precio 
     self.markup = Decimal(round(self.nivel_de_precio.factor - 1, 4)).quantize(Decimal("0.0000")) 
     self.utilidad = Decimal(self.nivel_de_precio.valor).quantize(Decimal("0.0000")) 
     self.total = Decimal(self.monto) * (1 + Decimal(self.markup)) 
     super(CotizacionDetalle, self).save(*args, **kwargs) 

    def __str__(self): 
     return str(self.descripcion) 

이 내 admin.py이다 (나는 또한 같은 결과 오류와 행동의 주석 버전을 시도했다). 이 변경에 대한 옵션을 FK 모델에서 선택할 수 있으면 좋겠지 만 마크 업 값을 .05 % 늘리고 마크 업을 .05 % 줄이려면 두 가지 조치를 취해야합니다. 어떤 경우에는 대신 마크 업 필드를 업데이트해야합니다. 업데이트

class CotizacionDetalleAdminForm(forms.ModelForm): 
    class Meta: 
     model = CotizacionDetalle 
     fields = ['item', 'cotizacion', 'cantidad', 'nivel_de_precio'] 


def cambiar_utilidad(modeladmin, request, queryset): 
    from .models import NivelDePrecio 
    ndp = NivelDePrecio.objects.get(id=5) 
    queryset.update(nivel_de_precio_id=ndp.id) 
    # for cd in queryset: 
    #  cd.nivel_de_precio_id = ndp.id 
    #  cd.save() 


cambiar_utilidad.short_description = "Cambiar Utilidad 25%" 


class CotizacionDetalleAdmin(admin.ModelAdmin): 
    # save_as = True 
    form = CotizacionDetalleAdminForm 
    list_display = ['cliente', 'itinerario', 'cotizacion', 'descripcion', 
        'cantidad', 'costo', 'monto', 'utilidad', 'markup', 'total'] 
    list_display_links = ['descripcion'] 
    readonly_fields = ['descripcion', 'costo', 'monto', 'utilidad', 'markup', 'total', 'slug', 'creado', 'actualizado'] 
    search_fields = ['descripcion'] 
    list_filter = (('cotizacion__itinerario__cliente', DropdownFilterRelated), 
        ('cotizacion__itinerario', DropdownFilterRelated), 
        ('cotizacion', DropdownFilterRelated),) 
    ordering = ['cotizacion__itinerario__cliente__codigo', 'cotizacion__fecha_ida', 'id'] 
    actions = [cambiar_utilidad] 


admin.site.register(CotizacionDetalle, CotizacionDetalleAdmin) 

:

는 ...이 시도 :

https://djangosnippets.org/snippets/1836/

admin.py

def create_action_nivel(nivel): 
    def action(modeladmin, request, queryset): 
     queryset.update(nivel_de_precio=nivel) 

    name = "change_to_%s" % (nivel.slug,) 
    return (name, (action, name, "Cambiar Nivel de Precio a: %s" % (nivel,))) 

에서 다음과 같이 ...

,
class CotizacionDetalleAdmin(admin.ModelAdmin): 

    def get_actions(self, request): 
     return dict(create_action_nivel(n) for n in NivelDePrecio.objects.all()) 

하지만이 같은 오류는, 역 추적은 여기에 표시 : 필드 업데이트 할 때 http://dpaste.com/360GEJY

같은 조각은 외래 키 필드를없는 잘 작동합니다. 나는 오류가 많은 시험 후, 문제가 설명 필드의 내용 안에 % 기호의 사용에서 발견

def create_action_estatus(estatus): 
    def action(modeladmin, request, queryset): 
     queryset.update(estatus=estatus) 

    name = "mark_%s" % (estatus,) 
    return name, (action, name, "Cambiar Estatus: %s " % (estatus,)) 


class ItinerarioAdmin(admin.ModelAdmin): 
    def get_actions(self, request): 
     statuses= [ "Solicitado", "Cotizado", "Confirmado", "Facturado", "Cerrado"] 
     return dict(create_action_estatus(e) for e in statuses) 

답변

0

: 나는 다른 모델에 그것을 사용하고 잘 작동합니다 분명히 관리자 작업을 만들 때 short_description 생성 된 문자열이 재사용되며 내부에 % 기호가 있으면 완료되지 않은 형식 오류가 발생합니다.

이 솔루션은이 같은 SHORT_DESCRIPTION 문자열을 구축하는 것이었다 다음 (탈출) 이중 % 기호를주의 ...

cambiar_nivel.short_description = "Cambiar Nivel de Precio a {0:.1f}%%".format(nivel.valor * 100) 

관련 문제