欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

[RoR] 实现一个auto_redirect_to

程序员文章站 2022-07-06 16:31:02
...
最近在使用RoR做项目,体会到了快速开发的乐趣,也遇到了一些困难,其中一个就是redirect_to。

我遇到的一个问题是,当使用Ajax更新页面局部内容时,session内容已经过期,这时需要整个页面跳转到登录页面。

直接调用redirect_to会使局部内容显示成登录页面,它是在HTTP头里写入重定向参数来实现的。在我这里的特殊情况下,正确的做法是让它执行一个包含在<script>标记中的脚本,在脚本中更改窗口location值来跳转。

不过RoR中使用Ajax时,会根据:update参数来决定是使用Updater还是Request。如果使用Updater方式,则应返回一段纯脚本;如果是Request方式,应返回一段包括在<script>标记中的脚本;如果是普通方式,就应该使用原有的redirect_to函数了。因为服务端无法区分使用的是哪种方式来请求,所以简单的做法是每个请求都附加一个参数用来区分,不加参数则是普通请求方式。

为了达到这个目的,我修改了prototype_helper中的remote_function函数。这个函数根据传递进来的参数来决定使用Request或是Updater,我就在这里下手:

<!---->      def remote_function(options)
        javascript_options 
= options_for_ajax(options)

        update 
= ''
        
if options[:update] and options[:update].is_a?Hash
          update  
= []
          update 
<< "success:'#{options[:update][:success]}'" if options[:update][:success]
          update 
<< "failure:'#{options[:update][:failure]}'" if options[:update][:failure]
          update  
= '{' + update.join(','+ '}'
        elsif options[:update]
          update 
<< "'#{options[:update]}'"
        end

        function 
= update.empty? ? 
          
"new Ajax.Request(" :
          
"new Ajax.Updater(#{update}, "
    
        url_options 
= options[:url]
        ajax_options 
= options[:update] ? {:ajax => 'update'} : {:ajax => 'request'}
        url_options 
=
 url_options.merge(ajax_options)
        url_options 
= url_options.merge(:escape => falseif url_options.is_a? Hash
        function 
<< "'#{url_for(url_options)}'"
        function 
<< ", #{javascript_options})"

        function 
= "#{options[:before]}; #{function}" if options[:before]
        function 
= "#{function}; #{options[:after]}"  if options[:after]
        function 
= "if (#{options[:condition]}) { #{function}; }" if options[:condition]
        function 
= "if (confirm('#{escape_javascript(options[:confirm])}')) { #{function}; }" if options[:confirm]

        
return function
      end

有红色的2行是我添加的,由于这个编辑器的原因,没有显示成整行红色。这2行的作用是判断是否有:update参数,用它来决定是添加ajax=update还是ajax=request。

现在可以实现一个简单的auto_redirect_to了:

<!---->  def auto_redirect_to(method, url)
    
case method
    when 
'request'
      request_redirect_to(url)
    when 
'update'
      update_redirect_to(url)
    
else
      redirect_to(url)
    end
  end
  
  def request_redirect_to(url)
    render :update 
do |page|
      page.redirect_to(url)
    end
  end
  
  def update_redirect_to(url)
    render :inline 
=> <<-EOS
      
<script language="javascript">
      
<%=
        render :update 
do |page|
          page.redirect_to(
"#{url_for(url)}")
        end
      
%>
      
</script>
    EOS
  end

使用helper方式使它能够被include到ApplicationController中就行了。

为了不和参数绑得太死,这里把method作为参数由调用者传入。

使用方法,以Login Engine为例,它在access_denied中处理跳转。在ApplicationController中重写这个函数:

<!---->    def access_denied
      auto_redirect_to(
params[:ajax], :controller => "/user", :action => "login")
      
false
    end  

现在可以测试了。请求可以是普通的(超链接),Updater方式(请求到一个DIV里),Request方式,现在都能够跳转到正确页面。

ajax参数通过hack库代码来实现,对于使用者来说基本上是透明的。