使用Rails 3和Jquery实现无限滚动

时间:2011-11-30 15:41:18

标签: ruby-on-rails jquery ruby-on-rails-3.1 infinite-scroll

我正在尝试在我的项目上实现无限滚动。我正在混合使用Railscast #114 Endless Pagethis

当我尝试在页面结束时停止发送请求时,除了奇怪的行为外,一切正常。

到目前为止,我有:

控制器:

def show
    @title = Photoset.find(params[:id]).name
    @photos = Photoset.find(params[:id]).photo.paginate(:page => params[:page], :per_page => 20)
    respond_to do |format|
      format.js
      format.html
    end
end

Show.html.erb:

<% content_for :body_class, '' %>
<%= render 'shared/header' %>
<div id="photos_container">
    <div id="photos_header">
    <h2><%= @title %></h2>
    </div>
    <%= render :partial => 'photo', :collection => @photos %>
</div>
<%= render :partial => 'endless_scroll' %>

Javascript(通过部分加载):

<script type="text/javascript">
(function() {
  var page = 1,
  loading = false,
  finish = false;

  function nearBottomOfPage() {
    return $(window).scrollTop() > $(document).height() - $(window).height() - 200;
  }

  function finish() {
    finish = true;
  }

  $(window).scroll(function(){
    if (loading) {
      return;
    }

    if(nearBottomOfPage() && !finish) {
      loading=true;
      page++;
      $.ajax({
        url: '/photosets/<%= params[:id] %>?page=' + page,
        type: 'get',
        dataType: 'script',
        success: function() {
          loading=false;
        }
      });
    }
  });
}());
</script>

show.js.erb

$("#photos_container").append("<%= escape_javascript(render :partial => 'photo', :collection => @photos) %>");
<% if @photos.total_pages == params[:page].to_i() %>
    page.call 'finish'
<% end %>

正如您所看到的,在我的show.js.erb上,我有一个page.call,它将finish变量赋予真。这会停止请求。

有线的事情是它永远不会加载最后一页。在@photos.total_pages == params[:page].to_i()而不是仅调用finish函数并将变量设置为true时,它还会阻止$("#photos_container").append("<%= escape_javascript(render :partial => 'photo', :collection => @photos) %>");运行。

它将请求发送到控制器,运行SQL但不附加最后一页。

如果我将条件更改为@photos.total_pages < params[:page].to_i()它可以正常工作,但会向不存在的页面发送额外请求。

我很感激我的实施方面的任何帮助。我不确定是否有更充分的(Rails)方法来实现这一目标。

1 个答案:

答案 0 :(得分:2)

首先,当请求为xhr类型时,您可以从部分渲染html:

def show
  photoset = Photoset.includes(:photos).find(params[:id])
  @title = photoset.name
  @photos = photoset.photo.paginate(:page => params[:page], :per_page => 20)
  if request.xhr?
    render '_photo', :layout => false
  end
end

然后使用ajax调用:

$.ajax({
  url: '/photosets/<%= params[:id] %>?page=' + page,
  type: 'get',
  dataType: 'script',
  success: function(response) {
    $("#photos_container").append(response);
    if (response == "") {
      //stop calling endless scroll
    }
  });  
});