mirror of
https://github.com/flarum/framework.git
synced 2025-01-10 13:03:43 +08:00
294 lines
11 KiB
JavaScript
294 lines
11 KiB
JavaScript
|
import Ember from 'ember';
|
||
|
|
||
|
var $ = Ember.$;
|
||
|
|
||
|
/**
|
||
|
Component which renders items in a `post-stream` object. It handles scroll
|
||
|
events so that when the user scrolls to the top/bottom of the page, more
|
||
|
posts will load. In doing this is also sends an action so that the parent
|
||
|
controller's state can be updated. Finally, it can be sent actions to jump
|
||
|
to a certain position in the stream and load posts there.
|
||
|
*/
|
||
|
export default Ember.Component.extend({
|
||
|
classNames: ['stream'],
|
||
|
|
||
|
// The stream object.
|
||
|
stream: null,
|
||
|
|
||
|
// Pause window scroll event listeners. This is set to true while loading
|
||
|
// posts, because we don't want a scroll event to trigger another block of
|
||
|
// posts to be loaded.
|
||
|
paused: false,
|
||
|
|
||
|
// Whether or not the stream's initial content has loaded.
|
||
|
loaded: Ember.computed.bool('stream.loadedCount'),
|
||
|
|
||
|
// When the stream content is not "active", window scroll event listeners
|
||
|
// will be ignored. For the stream content to be active, its initial
|
||
|
// content must be loaded and it must not be "paused".
|
||
|
active: Ember.computed('loaded', 'paused', function() {
|
||
|
return this.get('loaded') && !this.get('paused');
|
||
|
}),
|
||
|
|
||
|
// Whenever the stream object changes (i.e. we have transitioned to a
|
||
|
// different discussion), pause events and cancel any pending state updates.
|
||
|
refresh: Ember.observer('stream', function() {
|
||
|
this.set('paused', true);
|
||
|
clearTimeout(this.updateStateTimeout);
|
||
|
}),
|
||
|
|
||
|
didInsertElement: function() {
|
||
|
$(window).on('scroll', {view: this}, this.windowWasScrolled);
|
||
|
},
|
||
|
|
||
|
willDestroyElement: function() {
|
||
|
$(window).off('scroll', this.windowWasScrolled);
|
||
|
},
|
||
|
|
||
|
windowWasScrolled: function(event) {
|
||
|
event.data.view.update();
|
||
|
},
|
||
|
|
||
|
// Run any checks/updates according to the window's current scroll
|
||
|
// position. We check to see if any terminal 'gaps' are in the viewport
|
||
|
// and trigger their loading mechanism if they are. We also update the
|
||
|
// controller's 'start' query param with the current position. Note: this
|
||
|
// observes the 'active' property, so if the stream is 'unpaused', then an
|
||
|
// update will be triggered.
|
||
|
update: Ember.observer('active', function() {
|
||
|
if (!this.get('active')) { return; }
|
||
|
|
||
|
var $items = this.$().find('.item'),
|
||
|
$window = $(window),
|
||
|
marginTop = this.getMarginTop(),
|
||
|
scrollTop = $window.scrollTop() + marginTop,
|
||
|
viewportHeight = $window.height() - marginTop,
|
||
|
loadAheadDistance = 300,
|
||
|
startNumber,
|
||
|
endNumber;
|
||
|
|
||
|
// Loop through each of the items in the stream. An 'item' is either a
|
||
|
// single post or a 'gap' of one or more posts that haven't been
|
||
|
// loaded yet.
|
||
|
$items.each(function() {
|
||
|
var $this = $(this);
|
||
|
var top = $this.offset().top;
|
||
|
var height = $this.outerHeight(true);
|
||
|
|
||
|
// If this item is above the top of the viewport (plus a bit of
|
||
|
// leeway for loading-ahead gaps), skip to the next one. If it's
|
||
|
// below the bottom of the viewport, break out of the loop.
|
||
|
if (top + height < scrollTop - loadAheadDistance) { return; }
|
||
|
if (top > scrollTop + viewportHeight + loadAheadDistance) { return false; }
|
||
|
|
||
|
// If this item is a gap, then we may proceed to check if it's a
|
||
|
// *terminal* gap and trigger its loading mechanism.
|
||
|
if ($this.hasClass('gap')) {
|
||
|
var gapView = Ember.View.views[$this.attr('id')];
|
||
|
if ($this.is(':first-child')) {
|
||
|
gapView.set('direction', 'up').load();
|
||
|
} else if ($this.is(':last-child')) {
|
||
|
gapView.set('direction', 'down').load();
|
||
|
}
|
||
|
} else {
|
||
|
if (top + height < scrollTop + viewportHeight) {
|
||
|
endNumber = $this.data('number');
|
||
|
}
|
||
|
|
||
|
// Check if this item is in the viewport, minus the distance we
|
||
|
// allow for load-ahead gaps. If we haven't yet stored a post's
|
||
|
// number, then this item must be the FIRST item in the viewport.
|
||
|
// Therefore, we'll grab its post number so we can update the
|
||
|
// controller's state later.
|
||
|
if (top + height > scrollTop && ! startNumber) {
|
||
|
startNumber = $this.data('number');
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
|
||
|
// Finally, we want to update the controller's state with regards to the
|
||
|
// current viewing position of the discussion. However, we don't want to
|
||
|
// do this on every single scroll event as it will slow things down. So,
|
||
|
// let's do it at a minimum of 250ms by clearing and setting a timeout.
|
||
|
var view = this;
|
||
|
clearTimeout(this.updateStateTimeout);
|
||
|
this.updateStateTimeout = setTimeout(function() {
|
||
|
view.sendAction('positionChanged', startNumber || 1, endNumber);
|
||
|
}, 500);
|
||
|
}),
|
||
|
|
||
|
loadingNumber: function(number, noAnimation) {
|
||
|
// The post with this number is being loaded. We want to scroll to where
|
||
|
// we think it will appear. We may be scrolling to the edge of the page,
|
||
|
// but we don't want to trigger any terminal post gaps to load by doing
|
||
|
// that. So, we'll disable the window's scroll handler for now.
|
||
|
this.set('paused', true);
|
||
|
this.jumpToNumber(number, noAnimation);
|
||
|
},
|
||
|
|
||
|
loadedNumber: function(number, noAnimation) {
|
||
|
// The post with this number has been loaded. After we scroll to this
|
||
|
// post, we want to resume scroll events.
|
||
|
var view = this;
|
||
|
Ember.run.scheduleOnce('afterRender', function() {
|
||
|
view.jumpToNumber(number, noAnimation).done(function() {
|
||
|
view.set('paused', false);
|
||
|
});
|
||
|
});
|
||
|
},
|
||
|
|
||
|
loadingIndex: function(index, noAnimation) {
|
||
|
// The post at this index is being loaded. We want to scroll to where we
|
||
|
// think it will appear. We may be scrolling to the edge of the page,
|
||
|
// but we don't want to trigger any terminal post gaps to load by doing
|
||
|
// that. So, we'll disable the window's scroll handler for now.
|
||
|
this.set('paused', true);
|
||
|
this.jumpToIndex(index, noAnimation);
|
||
|
},
|
||
|
|
||
|
loadedIndex: function(index, noAnimation) {
|
||
|
// The post at this index has been loaded. After we scroll to this post,
|
||
|
// we want to resume scroll events.
|
||
|
var view = this;
|
||
|
Ember.run.scheduleOnce('afterRender', function() {
|
||
|
view.jumpToIndex(index, noAnimation).done(function() {
|
||
|
view.set('paused', false);
|
||
|
});
|
||
|
});
|
||
|
},
|
||
|
|
||
|
// Scroll down to a certain post by number (or the gap which we think the
|
||
|
// post is in) and highlight it.
|
||
|
jumpToNumber: function(number, noAnimation) {
|
||
|
// Clear the highlight class from all posts, and attempt to find and
|
||
|
// highlight a post with the specified number. However, we don't apply
|
||
|
// the highlight to the first post in the stream because it's pretty
|
||
|
// obvious that it's the top one.
|
||
|
var $item = this.$('.item').removeClass('highlight').filter('[data-number='+number+']');
|
||
|
if (!$item.is(':first-child')) {
|
||
|
$item.addClass('highlight');
|
||
|
}
|
||
|
|
||
|
// If we didn't have any luck, then a post with this number either
|
||
|
// doesn't exist, or it hasn't been loaded yet. We'll find the item
|
||
|
// that's closest to the post with this number and scroll to that
|
||
|
// instead.
|
||
|
if (!$item.length) {
|
||
|
$item = this.findNearestToNumber(number);
|
||
|
}
|
||
|
|
||
|
return this.scrollToItem($item, noAnimation);
|
||
|
},
|
||
|
|
||
|
// Scroll down to a certain post by index (or the gap the post is in.)
|
||
|
jumpToIndex: function(index, noAnimation) {
|
||
|
var $item = this.findNearestToIndex(index);
|
||
|
return this.scrollToItem($item, noAnimation);
|
||
|
},
|
||
|
|
||
|
scrollToItem: function($item, noAnimation) {
|
||
|
var $container = $('html, body').stop(true);
|
||
|
if ($item.length) {
|
||
|
var marginTop = this.getMarginTop();
|
||
|
var scrollTop = $item.is(':first-child') ? 0 : $item.offset().top - marginTop;
|
||
|
if (noAnimation) {
|
||
|
$container.scrollTop(scrollTop);
|
||
|
} else if (scrollTop !== $(document).scrollTop()) {
|
||
|
$container.animate({scrollTop: scrollTop});
|
||
|
}
|
||
|
}
|
||
|
return $container.promise();
|
||
|
},
|
||
|
|
||
|
// Find the DOM element of the item that is nearest to a post with a certain
|
||
|
// number. This will either be another post (if the requested post doesn't
|
||
|
// exist,) or a gap presumed to contain the requested post.
|
||
|
findNearestToNumber: function(number) {
|
||
|
var $nearestItem = $();
|
||
|
this.$('.item').each(function() {
|
||
|
var $this = $(this);
|
||
|
if ($this.data('number') > number) {
|
||
|
return false;
|
||
|
}
|
||
|
$nearestItem = $this;
|
||
|
});
|
||
|
return $nearestItem;
|
||
|
},
|
||
|
|
||
|
findNearestToIndex: function(index) {
|
||
|
var $nearestItem = this.$('.item[data-start='+index+'][data-end='+index+']');
|
||
|
if (! $nearestItem.length) {
|
||
|
this.$('.item').each(function() {
|
||
|
$nearestItem = $(this);
|
||
|
if ($nearestItem.data('end') >= index) {
|
||
|
return false;
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
return $nearestItem;
|
||
|
},
|
||
|
|
||
|
// Get the distance from the top of the viewport to the point at which we
|
||
|
// would consider a post to be the first one visible.
|
||
|
getMarginTop: function() {
|
||
|
return $('#header').outerHeight() + parseInt(this.$().css('margin-top'));
|
||
|
},
|
||
|
|
||
|
actions: {
|
||
|
goToNumber: function(number, noAnimation) {
|
||
|
number = Math.max(number, 1);
|
||
|
|
||
|
// Let's start by telling our listeners that we're going to load
|
||
|
// posts near this number. Elsewhere we will listen and
|
||
|
// consequently scroll down to the appropriate position.
|
||
|
this.trigger('loadingNumber', number, noAnimation);
|
||
|
|
||
|
// Now we have to actually make sure the posts around this new start
|
||
|
// position are loaded. We will tell our listeners when they are.
|
||
|
// Again, a listener will scroll down to the appropriate post.
|
||
|
var controller = this;
|
||
|
this.get('stream').loadNearNumber(number).then(function() {
|
||
|
controller.trigger('loadedNumber', number, noAnimation);
|
||
|
});
|
||
|
},
|
||
|
|
||
|
goToIndex: function(index, backwards, noAnimation) {
|
||
|
// Let's start by telling our listeners that we're going to load
|
||
|
// posts at this index. Elsewhere we will listen and consequently
|
||
|
// scroll down to the appropriate position.
|
||
|
this.trigger('loadingIndex', index, noAnimation);
|
||
|
|
||
|
// Now we have to actually make sure the posts around this index
|
||
|
// are loaded. We will tell our listeners when they are. Again, a
|
||
|
// listener will scroll down to the appropriate post.
|
||
|
var controller = this;
|
||
|
this.get('stream').loadNearIndex(index, backwards).then(function() {
|
||
|
controller.trigger('loadedIndex', index, noAnimation);
|
||
|
});
|
||
|
},
|
||
|
|
||
|
goToFirst: function() {
|
||
|
this.send('goToIndex', 0);
|
||
|
},
|
||
|
|
||
|
goToLast: function() {
|
||
|
this.send('goToIndex', this.get('stream.count') - 1, true);
|
||
|
|
||
|
// If the post stream is loading some new posts, then after it's
|
||
|
// done we'll want to immediately scroll down to the bottom of the
|
||
|
// page.
|
||
|
if (! this.get('stream.lastLoaded')) {
|
||
|
this.get('stream').one('postsLoaded', function() {
|
||
|
Ember.run.scheduleOnce('afterRender', function() {
|
||
|
$('html, body').stop(true).scrollTop($('body').height());
|
||
|
});
|
||
|
});
|
||
|
}
|
||
|
},
|
||
|
|
||
|
loadRange: function(start, end, backwards) {
|
||
|
this.get('stream').loadRange(start, end, backwards);
|
||
|
}
|
||
|
}
|
||
|
});
|