25

私はdjango-uniformを使用しており、いくつかの統一機能を使用するために、フォーム宣言から直接cssクラスを追加する方法を探しています(独立したウィジェット用)。

(おまけとして、ここに私の再利用可能な読み取り専用の自家製 mixin スニペットがあります...)

from django import forms

def _get_cleaner(form, field):
    def clean_field():
        return getattr(form.instance, field, None)
    return clean_field

class UniformROMixin(forms.BaseForm):
    """
    UniformROMixin, inherits to turn some fields read only

      - read_only = list of field names.
    """

    def __init__(self, *args, **kwargs):
        super(UniformROMixin, self).__init__(*args, **kwargs)
        if hasattr(self, "read_only"):
            if self.instance and self.instance.pk:
                for field in self.read_only:
                    self.fields[field].widget.attrs['readonly'] = True
                    self.fields[field].widget.attrs['class'] += "readOnly"
                    # here I would like to set css class of the label
                    # created from the self.fields[field].label string
                    setattr(self, "clean_" + field, _get_cleaner(self, field))

今のところ私の唯一の方法は、汎用フォーム テンプレートに JavaScript を少し追加し、手動でクラスを追加することです。

素晴らしいアイデアはありますか?

4

2 に答える 2

41

ウィジェットには、レンダリングする入力要素の属性を定義できるattrsキーワード引数があります。dictフォームには、Django がフォームを表示する方法を変更するために定義できるいくつかの属性もあります。次の例を見てください。

class MyForm(forms.Form):
    error_css_class = 'error'
    required_css_class = 'required'
    my_field = forms.CharField(max_length=10,
                               widget=forms.TextInput(attrs={'id': 'my_field',
                                                             'class': 'my_class'}))

これはどのWidgetクラスでも機能します。残念ながら、Django がラベルをレンダリングする方法を変更する簡単な方法はありません{{ field }}。幸いなことに、テンプレートでフォーム オブジェクトを少し操作します。

<form>
    {% for field in form %}
        <label class="my_class" for="{{ field.auto_id }}">{{ field.label }}</label>
        {{ field }}
    {% endfor %}
    <button type="submit">Submit</button>
</form>

繰り返しますが、作業中のオブジェクトにいつでも任意の属性を追加できます。

# In a view...
form = MyForm()
form.label_classes = ('class_a', 'class_b', )
# Or by hijacking ```__init__```
class MyForm(forms.Form):
    def __init__(self, *args, **kwargs):
        self.my_field = forms.CharField(max_length=10,
                                        widget=forms.TextInput(attrs={'id': 'my_field',
                                                                      'class': 'my_class'}))
        self.my_field.label_classes = ('class_a', 'class_b', )
        super(MyForm, self).__init__(*args, **kwargs)

コンテキスト内のフォームでテンプレートをレンダリングし、テンプレートで次のことができます。

<form>
    {% for field in form %}
        <label class="{% for class in field.label_classes %}{{ class }} {% endfor %}"
               for="{{ field.auto_id }}">{{ field.label }}</label>
        {{ field }}
    {% endfor %}
    <button type="submit">Submit</button>
</form>

あなたの空想に合うものは何でも。

于 2014-05-09T03:38:04.967 に答える
3

良い答えになるかもしれないこのスニペットを見つけました:

必須フィールドのラベルにcssクラスと「*」を追加する方法

ここで私のニーズに適応します(まだテストされていません。完了したら投稿を編集します):

from django.utils.html import escape

def readonly_cssclass_adder(bound_field, label_content, label_attrs):
    if 'readonly' in bound_field.field.widget.attrs:
        if 'class' in attrs:
            label_attrs['class'] += " readOnly"
        else:
            label_attrs['class'] = "readOnly"
    return label_content, label_attrs
        

def add_required_label_tag(original_function, tweak_foos=None):
    if not tweak_foos:
        return original_function

    def required_label_tag(self, contents=None, attrs=None):
        contents = contents or escape(self.label)
        if attrs is None:
            attrs = {}
        for foo in tweak_foos:
            contents, attrs = foo(self, contents, attrs)
        return original_function(self, contents, attrs)
    return required_label_tag

def decorate_bound_field():
    from django.forms.forms import BoundField
    BoundField.label_tag = add_required_label_tag(BoundField.label_tag, 
                                           tweak_foos=[readonly_cssclass_adder])

すべての BoundField クラスを微調整しないより良い解決策がある場合、私はまだ聞いています。

編集:必要なフィールドを処理するためのdjangoの統一された方法にリンクされている可能性がありますが、readonly_cssclass_adder. しかし、私は別のより簡単な解決策を見つけました。私の readOnly ウィジェットは自動的に readOnly ctrlHolder になりました

この追加は、今のところ私のお気に入りの応答です。

編集 2: 最後に選択したもう 1 つの方法は、uni_form/field.htmlBoundField.label_tag を呼び出さないテンプレートを「オーバーライド」することでした。そこで、ここでフィールドの状態を確認しました。

<label for="{{ field.auto_id }}"{% if field.field.required %}
       class="requiredField{% if field.widget.attrs.readonly %} readOnlyLabel{% endif %}"
       {% else %}{% if field.widget.attrs.readonly %}class="readOnlyLabel"{% endif %}{% endif %}>
    {{ field.label|safe }}{% if field.field.required %}<span class="asteriskField">*</span>{% endif %}
</label>
于 2011-08-06T15:07:13.317 に答える