0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-01-20 22:42:53 -05:00

Infinite Scroll Pagination for content screen

Fixes #258

- Modified post collection to have default values for paging.
- Added scroll handler to content view to check for more posts and load
as appropriate.
- Sanitized result from server-side post paging, ensure page # is
returned as an integer.
- Added a functional test stub.
This commit is contained in:
William Dibbern 2013-09-15 18:34:23 -05:00
parent 07629dd9ab
commit 4b8806ec1d
4 changed files with 76 additions and 1 deletions

View file

@ -44,8 +44,15 @@
});
Ghost.Collections.Posts = Backbone.Collection.extend({
currentPage: 1,
totalPages: 0,
totalPosts: 0,
nextPage: 0,
prevPage: 0,
url: Ghost.settings.apiRoot + '/posts',
model: Ghost.Models.Post,
parse: function (resp) {
if (_.isArray(resp.posts)) {
this.limit = resp.limit;

View file

@ -20,6 +20,8 @@
// -----------------------
ContentList = Ghost.View.extend({
isLoading: false,
events: {
'click .content-list-content' : 'scrollHandler'
},
@ -27,15 +29,67 @@
initialize: function (options) {
this.$('.content-list-content').scrollClass({target: '.content-list', offset: 10});
this.listenTo(this.collection, 'remove', this.showNext);
// Can't use backbone event bind (see: http://stackoverflow.com/questions/13480843/backbone-scroll-event-not-firing)
this.$('.content-list-content').scroll($.proxy(this.checkScroll, this));
},
showNext: function () {
if (this.isLoading) { return; }
var id = this.collection.at(0).id;
if (id) {
Backbone.trigger('blog:activeItem', id);
}
},
reportLoadError: function (response) {
var message = 'A problem was encountered while loading more posts';
if (response) {
// Get message from response
message += '; ' + Ghost.Views.Utils.getRequestErrorMessage(response);
} else {
message += '.';
}
Ghost.notifications.addItem({
type: 'error',
message: message,
status: 'passive'
});
},
checkScroll: function (event) {
var self = this,
element = event.target,
triggerPoint = 100;
// If we haven't passed our threshold, exit
if (this.isLoading || (element.scrollTop + element.clientHeight + triggerPoint <= element.scrollHeight)) {
return;
}
// If we've loaded the max number of pages, exit
if (this.collection.currentPage >= this.collection.totalPages) {
return;
}
// Load moar posts!
this.isLoading = true;
this.collection.fetch({
data: {
status: 'all',
page: (self.collection.currentPage + 1),
orderBy: ['updated_at', 'DESC']
}
}).then(function onSuccess(response) {
self.render();
self.isLoading = false;
}, function onError(e) {
self.reportLoadError(e);
});
},
render: function () {
this.collection.each(function (model) {
this.$('ol').append(this.addSubview(new ContentItem({model: model})).render().el);

View file

@ -259,7 +259,7 @@ Post = GhostBookshelf.Model.extend({
var totalPosts = resp[0].aggregate,
data = {
posts: collection.toJSON(),
page: opts.page,
page: parseInt(opts.page, 10),
limit: opts.limit,
pages: Math.ceil(totalPosts / opts.limit),
total: totalPosts

View file

@ -44,6 +44,20 @@ casper.test.begin("Content screen is correct", 17, function suite(test) {
// casper.clickLabel(testPost.title, "h3");
// });
casper.run(function () {
test.done();
});
});
casper.test.begin('Infinite scrolling', 1, function suite(test) {
test.filename = 'content_infinite_scrolling_test.png';
// Placeholder for infinite scrolling/pagination tests (will need to setup 16+ posts).
casper.start(url + "ghost/content/", function testTitleAndUrl() {
test.assertTitle("", "Ghost admin has no title");
}).viewport(1280, 1024);
casper.run(function () {
test.done();
});