19

我有一个设置为绑定到的 div observeableArray,但我只想observeableArray在任何给定时间显示最多 50 个项目。我想通过页面上的上一个和下一个按钮以及索引来处理这个问题,以允许用户循环浏览集合中的项目页面。
我知道我可能可以使用computedObservable自定义数据绑定来做到这一点,但我不知道该怎么做(我仍然是淘汰赛新手)。
谁能指出我正确的方向?

这是我的代码(JS 在 TypeScript 中):

<div class="container-fluid">
    <div class="row-fluid">
        <div class="span12">
            <%=
            if params[:q]
              render 'active_search.html.erb'
            else
              render 'passive_search.html.erb'
            end
            %>
            <%= form_tag("/search", method: "get", :class => "form-search form-inline") do %>
            <%= label_tag(:q, "Search for:") %>
            <%= text_field_tag(:q, nil, class:"input-medium search-query") %>
            <%= submit_tag("Search", :class=>"btn") %>
            <% end %>

            <div class="media" data-bind="foreach: tweetsArray">
                <%= image_tag('twitter-icon.svg', :class=>"tweet_img", :style=>"display:inline;") %>
                <div class="media-body" style="display:inline;">
                    <h4 class="media-heading" data-bind="text: user.screen_name" style="display:inline;"></h4>
                    <span data-bind="text:text" style="display:inline;"></span> <br />
                    <span data-bind="text:'Created at '+created_at"></span> <br />
                </div>
            </div>

            <div class="pagination pagination-centered">
                <ul>
                    <li>
                        <a href="#">Prev</a>
                    </li>
                    <li>
                        <a href="#">1</a>
                    </li>
                    <li>
                        <a href="#">Next</a>
                    </li>
                </ul>
            </div>

        </div>
    </div>
</div>

<script>
    var viewModel = new twitterResearch.TweetViewModel();
    ko.applyBindings(viewModel);

    //TODO: notes to self, use custom binding for pagination along with a computed observable to determine where at in the list you are

    //document.onReady callback function
    $(function() {
        $.getJSON('twitter', {}, function(data) {
            viewModel.pushTweet(data);
            console.log(data.user);
        });
    });
</script>

declare var $: any;
declare var ko: any;

module twitterResearch {
    class Tweet {
        text: string;
        created_at: string;
        coordinates: string;
        user: string;
        entities: string;
        id: number;
        id_str: string;

        constructor(_text: string, _created_at: string, _coordinates: any, _user: any,
                    _entities: any, _id_str: string, _id: number){

            this.text = _text;
            this.created_at = _created_at;
            this.coordinates = _coordinates;
            this.user = _user;
            this.entities = _entities;
            this.id_str = _id_str;
            this.id = _id;
        }
    }

    export class TweetViewModel{

        tweetsArray: any;
        constructor()
        {
            this.tweetsArray = ko.observableArray([]);
        }

        //tweet is going to be the JSON tweet we return
        //from the server
        pushTweet(tweet)
        {
            var _tweet = new Tweet(tweet.text, tweet.created_at, tweet.coordinates,
                                    tweet.user, tweet.entities, tweet.id_str, tweet.id);
            this.tweetsArray.push(_tweet);
            this.tweetsArray.valueHasMutated();
        }
    }
}
4

4 回答 4

43

使用 Knockout 进行分页非常简单。我个人会以这种方式实现它:

  • 有一个包含所有元素的 observableArray
  • 有一个包含当前页面的可观察对象(初始化为 0)
  • 有一个变量来声明每页的元素数量
  • 有一个返回页数的计算,由于每页的元素数和元素总数而计算。
  • 最后,添加一个对包含所有元素的数组进行切片的计算。

鉴于此,您现在可以添加一个增加(下一个)或减少(上一个)当前页面的函数。

这是一个简单的例子:

var Model = function() {
    var self = this;
    this.all = ko.observableArray([]);
    this.pageNumber = ko.observable(0);
    this.nbPerPage = 25;
    this.totalPages = ko.computed(function() {
        var div = Math.floor(self.all().length / self.nbPerPage);
        div += self.all().length % self.nbPerPage > 0 ? 1 : 0;
        return div - 1;
    });

    this.paginated = ko.computed(function() {
        var first = self.pageNumber() * self.nbPerPage;
        return self.all.slice(first, first + self.nbPerPage);
    });

    this.hasPrevious = ko.computed(function() {
        return self.pageNumber() !== 0;
    });

    this.hasNext = ko.computed(function() {
        return self.pageNumber() !== self.totalPages();
    });

    this.next = function() {
        if(self.pageNumber() < self.totalPages()) {
            self.pageNumber(self.pageNumber() + 1);
        }
    }

    this.previous = function() {
        if(self.pageNumber() != 0) {
            self.pageNumber(self.pageNumber() - 1);
        }
    }
}

您会在这里找到一个简单而完整的示例:http: //jsfiddle.net/LAbCv/(可能有点错误,但想法就在那里)。

于 2013-07-16T07:21:16.500 回答
9

实际上我正在一个网站上工作,该网站有很多表格(其中大多数需要分页)。
所以实际上,我需要一些reusable-component分页来在我需要分页的所有情况下使用它。
此外,我需要比这个问题的公认答案中提供的更高级的功能。

所以我开发了自己的组件来解决这个问题,就是这样。

现在在 Github

提琴手

有关更多详细信息,请继续阅读(请考虑从 GitHub 获取代码,而不是从这里获取,因为 GitHub 代码在我放在这里后已经更新和增强)

JavaScript

function PagingVM(options) {
    var self = this;

    self.PageSize = ko.observable(options.pageSize);
    self.CurrentPage = ko.observable(1);
    self.TotalCount = ko.observable(options.totalCount);

    self.PageCount = ko.pureComputed(function () {
        return Math.ceil(self.TotalCount() / self.PageSize());
    });

    self.SetCurrentPage = function (page) {
        if (page < self.FirstPage)
            page = self.FirstPage;

        if (page > self.LastPage())
            page = self.LastPage();

        self.CurrentPage(page);
    };

    self.FirstPage = 1;
    self.LastPage = ko.pureComputed(function () {
        return self.PageCount();
    });

    self.NextPage = ko.pureComputed(function () {
        var next = self.CurrentPage() + 1;
        if (next > self.LastPage())
            return null;
        return next;
    });

    self.PreviousPage = ko.pureComputed(function () {
        var previous = self.CurrentPage() - 1;
        if (previous < self.FirstPage)
            return null;
        return previous;
    });

    self.NeedPaging = ko.pureComputed(function () {
        return self.PageCount() > 1;
    });

    self.NextPageActive = ko.pureComputed(function () {
        return self.NextPage() != null;
    });

    self.PreviousPageActive = ko.pureComputed(function () {
        return self.PreviousPage() != null;
    });

    self.LastPageActive = ko.pureComputed(function () {
        return (self.LastPage() != self.CurrentPage());
    });

    self.FirstPageActive = ko.pureComputed(function () {
        return (self.FirstPage != self.CurrentPage());
    });

    // this should be odd number always
    var maxPageCount = 7;

    self.generateAllPages = function () {
        var pages = [];
        for (var i = self.FirstPage; i <= self.LastPage() ; i++)
            pages.push(i);

        return pages;
    };

    self.generateMaxPage = function () {
        var current = self.CurrentPage();
        var pageCount = self.PageCount();
        var first = self.FirstPage;

        var upperLimit = current + parseInt((maxPageCount - 1) / 2);
        var downLimit = current - parseInt((maxPageCount - 1) / 2);

        while (upperLimit > pageCount) {
            upperLimit--;
            if (downLimit > first)
                downLimit--;
        }

        while (downLimit < first) {
            downLimit++;
            if (upperLimit < pageCount)
                upperLimit++;
        }

        var pages = [];
        for (var i = downLimit; i <= upperLimit; i++) {
            pages.push(i);
        }
        return pages;
    };

    self.GetPages = ko.pureComputed(function () {
        self.CurrentPage();
        self.TotalCount();

        if (self.PageCount() <= maxPageCount) {
            return ko.observableArray(self.generateAllPages());
        } else {
            return ko.observableArray(self.generateMaxPage());
        }
    });

    self.Update = function (e) {
        self.TotalCount(e.TotalCount);
        self.PageSize(e.PageSize);
        self.SetCurrentPage(e.CurrentPage);
    };

    self.GoToPage = function (page) {
        if (page >= self.FirstPage && page <= self.LastPage())
            self.SetCurrentPage(page);
    }

    self.GoToFirst = function () {
        self.SetCurrentPage(self.FirstPage);
    };

    self.GoToPrevious = function () {
        var previous = self.PreviousPage();
        if (previous != null)
            self.SetCurrentPage(previous);
    };

    self.GoToNext = function () {
        var next = self.NextPage();
        if (next != null)
            self.SetCurrentPage(next);
    };

    self.GoToLast = function () {
        self.SetCurrentPage(self.LastPage());
    };
}

HTML

<ul data-bind="visible: NeedPaging" class="pagination pagination-sm">
    <li data-bind="css: { disabled: !FirstPageActive() }">
        <a data-bind="click: GoToFirst">First</a>
    </li>
    <li data-bind="css: { disabled: !PreviousPageActive() }">
        <a data-bind="click: GoToPrevious">Previous</a>
    </li>

    <!-- ko foreach: GetPages() -->
    <li data-bind="css: { active: $parent.CurrentPage() === $data }">
        <a data-bind="click: $parent.GoToPage, text: $data"></a>
    </li>
    <!-- /ko -->

    <li data-bind="css: { disabled: !NextPageActive() }">
        <a data-bind="click: GoToNext">Next</a>
    </li>
    <li data-bind="css: { disabled: !LastPageActive() }">
        <a data-bind="click: GoToLast">Last</a>
    </li>
</ul>

特征

  1. 按需显示
    当根本不需要分页时(例如需要显示小于页面大小的项目),则HTML组件将消失。
    这将由语句建立data-bind="visible: NeedPaging"

  2. 根据需要禁用
    ,例如,如果您已经选择了最后一页,为什么应该可以按下last page或按钮?我正在处理这个问题,在这种情况下,我通过应用以下绑定来禁用这些按钮Next
    data-bind="css: { disabled: !PreviousPageActive() }"

  3. 区分所选页面 在所选页面上
    应用 一个特殊的类(在本例中称为active类),以使用户知道他/她现在在哪个页面中。
    这是由绑定建立的data-bind="css: { active: $parent.CurrentPage() === $data }"

  4. 最后和第一
    页也可以通过专用于此的简单按钮进入第一页和最后一页。

  5. 显示按钮的限制
    假设您有很多页面,例如 1000 页,那么会发生什么?你会为用户显示它们吗?绝对不是你必须根据当前页面显示其中的几个。例如,显示所选页面之前的 3 页和之后的其他 3 页。
    这种情况已经在这里处理,<!-- ko foreach: GetPages() -->
    GetPages函数应用一个简单的算法来确定我们是否需要显示所有页面(页面计数低于阈值,这可以很容易地确定),或者只显示一些按钮。您可以通过更改变量
    的值来确定阈值 现在我将其分配如下maxPageCount
    var maxPageCount = 7;这意味着不能为用户显示超过 7 个按钮(SelectedPage 之前 3 个,Selected Page 之后 3 个)和 Selected Page 本身。您可能想知道,如果在当前页面之后或

    之前 没有足够的页面来显示怎么办?别担心,我在算法中处理这个,例如,如果你有并且你有和当前的,那么接下来的页面将被显示 ,所以我们总是分层,在前面的例子中显示所选页面之前的页面和之后的页面选定的页面。
    11 pagesmaxPageCount = 7selected page is 10

    5,6,7,8,9,10(selected page),11

    maxPageCount51

  6. Selected Page Validation 对observable
    的所有设置操作,这些操作由用户决定选择的页面,正在执行该函数。仅在这个函数中我们设置了这个 observable,从代码中可以看出,在设置值之前我们进行验证操作以确保我们不会超出页面的可用页面。CurrentPageSetCurrentPage

  7. 已经干净
    了,我只使用pureComputed不使用computed属性,这意味着您无需为清洁和处理这些属性而烦恼。虽然,正如您将在下面的示例中看到的那样,您需要处理组件本身之外的一些其他订阅

注意 1
你可能会注意到我bootstrap在这个组件中使用了一些类,这很适合我,但是,当然,你可以使用自己的类而不是引导类。
我在这里使用的引导类是pagination、和pagination-sm, 可以根据需要随意更改它们。activedisabled

NOTE 2
所以我为你介绍了这个组件,是时候看看它是如何工作的了。
您可以像这样将此组件集成到您的主 ViewModel 中。

function MainVM() {
    var self = this;

    self.PagingComponent = ko.observable(new Paging({
        pageSize: 10,      // how many items you would show in one page
        totalCount: 100,   // how many ALL the items do you have.
    }));

    self.currentPageSubscription = self.PagingComponent().CurrentPage.subscribe(function (newPage) {
        // here is the code which will be executed when the user changes the page.
        // you can handle this in the way you need.
        // for example, in my case, I am requesting the data from the server again by making an ajax request
        // and then updating the component

        var data = /*bring data from server , for example*/
        self.PagingComponent().Update({

            // we need to set this again, why? because we could apply some other search criteria in the bringing data from the server, 
            // so the total count of all the items could change, and this will affect the paging
            TotalCount: data.TotalCount,

            // in most cases we will not change the PageSize after we bring data from the server
            // but the component allows us to do that.
            PageSize: self.PagingComponent().PageSize(),

            // use this statement for now as it is, or you have to made some modifications on the 'Update' function.
            CurrentPage: self.PagingComponent().CurrentPage(),
        });
    });

    self.dispose = function () {
        // you need to dispose the manual created subscription, you have created before.
        self.currentPageSubscription.dispose();
    }
}

最后但并非最不重要的一点,当然不要忘记根据您的特殊 viewModel 更改 HTML 组件中的绑定,或者用with binding类似这样的包装所有组件

<div data-bind="with: PagingComponent()">
    <!-- put the component here -->
</div>

干杯

于 2017-06-10T07:50:43.523 回答
1

我创建了一篇博文,详细解释了如何借助一个小 JQuery 插件(这里)创建分页。

基本上,我使用 AJAX 的普通敲除数据绑定,从服务器检索数据后,我调用插件。你可以在这里找到插件。它被称为简单分页

于 2014-02-20T08:23:48.600 回答
1

这个问题仍然是“knockout pagination”的热搜之一,所以knockout 扩展knockout-paging ( git ) 值得一提。
它通过扩展提供分页ko.observableArray。它有据可查且易于使用。
用法示例在这里

于 2017-10-23T19:39:44.387 回答