【问题标题】:Django datetimefield not showing datetimepicker widget for new inline form rowsDjango datetimefield 未显示新内联表单行的 datetimepicker 小部件
【发布时间】:2020-05-27 19:54:45
【问题描述】:

关于下图,当我点击“添加条目”时,新条目行不再有 datetimepicker 小部件。这很奇怪,因为它有正确的类。 +1 字段代表 django 表单自动添加的附加字段。这似乎工作正常。只有在页面呈现后添加的字段可能是解决方案的关键。

我使用 flatpickr 包作为我的 datetimepicker 小部件。

trade_form.html

{% extends 'dashboard/base.html' %}
{% load static %}

{% block content %}

<script src="//ajax.googleapis.com/ajax/libs/jquery/2.1.3/jquery.min.js"></script>
<script src="{% static 'js/formset/jquery.formset.js' %}"></script>

<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/flatpickr/dist/flatpickr.min.css">
<script src="https://cdn.jsdelivr.net/npm/flatpickr"></script>


<!-- Page Heading -->
<div class="d-sm-flex align-items-center justify-content-between mb-3">
    <h2>New Trade</h2>
</div>

<div class="row">
    <div class="col">
        <form action="" method="post">{% csrf_token %}
            {{ form.media }}
            {{ form.as_p }}

            <table class="table">
                {{ entries.management_form }}

                {% for form in entries.forms %}
                {% if forloop.first %}
                <thead>
                <tr>
                    {% for field in form.visible_fields %}
                    <th>{{ field.label|capfirst }}</th>
                    {% endfor %}
                </tr>
                </thead>
                {% endif %}
                <tr class="{% cycle row1 row2 %} formset_row">
                    {% for field in form.visible_fields %}
                    <td>
                        {# Include the hidden fields in the form #}
                        {% if forloop.first %}
                        {% for hidden in form.hidden_fields %}
                        {{ hidden }}
                        {% endfor %}
                        {% endif %}
                        {{ field.errors.as_ul }}
                        {{ field }}
                    </td>
                    {% endfor %}
                </tr>
                {% endfor %}
            </table>
            <input type="submit" value="Save"/> <a href="{% url 'trade-list' %}">back to the list</a>
        </form>
    </div>


</div>

<script type="text/javascript">
    $('.formset_row').formset({
        addText: 'add entry',
        deleteText: 'remove',
        prefix: 'entry_set'
    });

</script>

<!-- Makes DateTimeFields Show Picker-->
<script>
    // This code activates flatpickr on fields with the 'datetimefield' class when the document has loaded
    window.addEventListener("DOMContentLoaded", function () {
        flatpickr(".datetimefield", {
            enableTime: true,
            enableSeconds: true,
            dateFormat: "Y-m-d H:i:S",
        });
    });
</script>

{% endblock content %}

forms.py

class EntriesForm(ModelForm):
    class Meta:
        model = Entry
        exclude = ()
        widgets = {
            'date': forms.DateTimeInput(format='%Y-%m-%d %H:%M:%S', attrs={'class': 'datetimefield'}),
        }

    # Adding default class to every visible field 

    def __init__(self, *args, **kwargs):
        super(EntriesForm, self).__init__(*args, **kwargs)
        for field_name, field in self.fields.items():
            if field.widget.attrs.get('class'):
                field.widget.attrs['class'] += ' form-control'
            else:
                field.widget.attrs['class'] = 'form-control'

EntriesFormSet = inlineformset_factory(Trade, Entry, form=EntriesForm, extra=1)

更新了 trade_form.html 脚本,但仍然没有成功。 到目前为止,这段代码只是删除了已经工作的字段上的 datetimepicker 小部件。所以可能只是格式问题。

<!-- Makes DateTimeFields Show Picker-->
<script>
    // This code activates flatpickr on fields with the 'datetimefield' class when the document has loaded
    window.addEventListener(

        "DOMContentLoaded", function () {
            flatpickr(".datetimefield", {
                enableTime: true,
                enableSeconds: true,
                dateFormat: "Y-m-d H:i:S",
            });
        }

        "added", function() {
            flatpickr(".datetimefield", {
                enableTime: true,
                enableSeconds: true,
                dateFormat: "Y-m-d H:i:S",
            });

        }

    );
</script>

【问题讨论】:

    标签: javascript django django-forms datetimepicker flatpickr


    【解决方案1】:

    added 需要放在的地方是在django-dynamic-formset 调用中,而不是在addEventListener 代码中,因为调用是由django-dynamic-formset 自己完成的。

    那是:(注意:我自己没有测试过,因为我现在正在打电话)

    <script type="text/javascript">
    
        function applyFlatpickr(parent) {
         flatpickr(parent.find(".datetimefield"), {
                    enableTime: true,
                    enableSeconds: true,
                    dateFormat: "Y-m-d H:i:S",
                });
        }
        <!-- Makes DateTimeFields Show Picker-->
            // This code activates flatpickr on fields with the 'datetimefield' class when the document has loaded
            window.addEventListener("DOMContentLoaded", function(){applyFlatpickr($(document))});
    
            $('.formset_row').formset({
                addText: 'add entry',
                deleteText: 'remove',
                prefix: 'entry_set',
                added: applyFlatpickr
            });
        </script>
    </script>
    

    django-dynamic-formset 将新创建的行作为 jQuery 对象传递给 added 函数,因此通过在参数上使用.find,我们可以仅匹配新添加的行并将 flatpickr 仅应用于该新行。

    来自django-dynamic-formset 文档:

    added 如果将此设置为一个函数,则该函数将在每个函数中调用 添加新表单的时间。该函数应采用单个参数, row;它将被传递一个 jQuery 对象,包装表单 刚刚添加。)

    【讨论】:

    • 很好,这完全有道理,但现在有一个新问题。当我单击“添加条目”时,已经存在的字段被更新。以下是概述问题的 3 张图片:imgur.com/a/1BCfVyL
    • @AlexWinkler 我已经在答案中更新了这个,所以当点击add 按钮时它不应该清除所有字段;这能解决问题吗?
    • 嘿@SimeonJM,传奇!是的,它奏效了。与父母的有趣方法。我仍然对它为什么起作用感到完全困惑,但会继续努力解决它。
    • @AlexWinkler 基本的事情是:flatpickr 将应用于作为第一个参数传递的任何元素列表。在DOMContentLoaded 事件中,$(document) 被传入,parent.find(".datetimefield") 将在页面上找到具有datetimefield 类的所有字段。添加新行时,将有效使用row.find(".datetimefield")(其中rowdjango-dynamic-formset 新添加的行)。这意味着flatpickr 将仅适用于具有datetimefield 类的字段在新添加的行内,并且已经存在的行不受影响。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2018-03-25
    • 2018-01-29
    • 1970-01-01
    • 2019-03-02
    • 1970-01-01
    • 2022-01-03
    • 1970-01-01
    相关资源
    最近更新 更多