繁体   English   中英

Django:如何在过滤后刷新表格而不刷新整个页面?

[英]Django: How to refresh table after filtering without refreshing the whole page?

问题的根源在于如何在不刷新整个页面的情况下动态刷新过滤的表。

我几乎是python / html / css的新手,所以请像对待新手一样发表一些意见[谢谢]。 经过对StackOverFlow的研究,我发现它可以用js来制作,但是我几乎没有js的经验,我也不知道如何在Django中使用它。

仅使用Django工具有可能吗? 效率如何? 也许您可以提供一些解决此问题的示例。

这是模型:

class Player(models.Model):
    last_name = models.CharField(
        null=True,
        blank=True,
        max_length=255,
        verbose_name="прізвище"
    )
    first_name = models.CharField(
        null=True,
        blank=True,
        max_length=255,
        verbose_name="ім'я"
    )
    city = models.ForeignKey(
        City,
        on_delete=models.SET_NULL,
        null=True,
        blank=True,
        verbose_name="місто"
    )
    rating = models.PositiveIntegerField(
        null=True,
        blank=True,
        verbose_name="рейтинг"
    )
    rank = models.ForeignKey(
        Rank,
        on_delete=models.SET_NULL,
        null=True,
        blank=True,
        verbose_name="ранг"
    )
    local_rank = models.ForeignKey(
        LocalRank,
        null=True,
        blank=True,
        on_delete=models.SET_NULL,
        verbose_name="розряд"
    )

def __str__(self):
    if self.last_name and self.first_name:
        return self.last_name + ' ' + self.first_name
    elif self.egd_last_name and self.egd_first_name:
        return self.egd_last_name + ' ' + self.egd_first_name
    else:
        return self.id

我正在使用django-tables2渲染表:

class PlayerTable(tables.Table):
    full_name = tables.LinkColumn(
        accessor="__str__",
        verbose_name="Прізвище та ім'я",
        order_by="last_name",
        viewname='UGD:player_info',
        empty_values=(),
        args=[A('pk')]
    )
    local_rank = tables.Column(
        accessor="local_rank.abbreviate",
        order_by="id"
    )
    ufgo_member = tables.BooleanColumn(
        verbose_name="Член УФГО"
    )

    class Meta:
        model = Player
        fields = (
            'id',
            'full_name',
            'city',
            'rating',
            'rank',
            'local_rank',
            'ufgo_member'
        )
        attrs = {'class': 'main'}

我正在使用django-filter制作过滤器形式:

    class PlayersFilter(django_filters.FilterSet):
    last_name = django_filters.CharFilter(
        lookup_expr='contains',
        label="Прізвище"
    )
    first_name = django_filters.CharFilter(
        lookup_expr='contains',
        label="Ім'я"
    )
    city = django_filters.ChoiceFilter(
        choices=[(city.id, city.name) for city in City.objects.all()],
        empty_label="--Не обрано--",
        label="Місто"
    )
    ufgo_member = django_filters.ChoiceFilter(
        choices=[
            (False, 'Ні'),
            (True, 'Так')
        ],
        name="ufgo_member",
        label="Член УФГО",
    )

    class Meta:
        model = Player
        fields = (
            'last_name',
            'first_name'
        )

以下是问题-视图/模板。

我在视图中使用SingleTableMixinFilterView来处理表和过滤器:

class RatingListView(SingleTableMixin, FilterView):
    table_class = PlayerTable
    table_pagination = False
    template_name = 'UGD/rating_list.html'
    filterset_class = PlayersFilter

在这里,我有一个将视图分为几个部分的想法,但是我仍然不知道该怎么做。 也许您有一些改善建议?

我的模板:

<body>
    <div class="filter">
        <form id="filter_submit" class="filter">
            {% block content %}
                <div class="filter">
                    <table class="filter">
                        {{ filter.form.as_table }}
                    </table>
                    <button id="filter_submit_button" type="submit">OK</button>
                </div>
            {% endblock %}
        </form>
    </div>
    <div>
        {% render_table table %}
    </div>
</body>

我想我应该添加一些脚本,但是我还不知道如何使用js

问题:选择过滤器并按“确定”后,整个页面将用新数据刷新

我希望只刷新表。

请给我一个关于如何做的提示。

非常感谢你。

查看django-filter文档( https://django-filter.readthedocs.io/en/develop/index.html ),此应用程序在后端进行过滤。 我没有看到执行此操作的任何JavaScript代码,它可以处理视图中的过滤逻辑。 因此,该应用程序需要刷新页面以向您显示过滤后的表。

我将建议您使用javascript实现过滤功能。 您需要确定要过滤的表的元素,触发器元素(下拉菜单和搜索框,如django-filter所示),并将它们链接在一起以对DOM进行修改。 这样,您可以拥有一个动态表过滤器。

这些链接可以帮助您开始:

http://www.w3schools.com/howto/howto_js_filter_table.asp

http://codepen.io/abocati/pen/vdKce [1]

[1] JS

(function(document) {
'use strict';

var LightTableFilter = (function(Arr) {

    var _input;
var _select;

    function _onInputEvent(e) {
        _input = e.target;
        var tables = document.getElementsByClassName(_input.getAttribute('data-table'));
        Arr.forEach.call(tables, function(table) {
            Arr.forEach.call(table.tBodies, function(tbody) {
                Arr.forEach.call(tbody.rows, _filter);
            });
        });
    }

    function _onSelectEvent(e) {
        _select = e.target;
        var tables = document.getElementsByClassName(_select.getAttribute('data-table'));
        Arr.forEach.call(tables, function(table) {
            Arr.forEach.call(table.tBodies, function(tbody) {
                Arr.forEach.call(tbody.rows, _filterSelect);
            });
        });
    }

    function _filter(row) {

        var text = row.textContent.toLowerCase(), val = _input.value.toLowerCase();
        row.style.display = text.indexOf(val) === -1 ? 'none' : 'table-row';

    }

    function _filterSelect(row) {

        var text_select = row.textContent.toLowerCase(), val_select = _select.options[_select.selectedIndex].value.toLowerCase();
        row.style.display = text_select.indexOf(val_select) === -1 ? 'none' : 'table-row';

    }

    return {
        init: function() {
            var inputs = document.getElementsByClassName('light-table-filter');
            var selects = document.getElementsByClassName('select-table-filter');
            Arr.forEach.call(inputs, function(input) {
                input.oninput = _onInputEvent;
            });
            Arr.forEach.call(selects, function(select) {
     select.onchange  = _onSelectEvent;
            });
        }
    };
})(Array.prototype);

document.addEventListener('readystatechange', function() {
    if (document.readyState === 'complete') {
        LightTableFilter.init();
    }
});

})(document);

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM