Rails源码研究之ActionView:五,prototype_helper

Rails的Ajax是依赖于prototype库的,我们来看看ActionView对prototype的一些方法调用的封装
源文件为prototype_helper.rb,其中包括RJS的update_page调用的JavaScriptGenerator类的定义:
1,link_to_remote是简单的异步url请求
def link_to_remote(name, options = {}, html_options = {})  
  link_to_function(name, remote_function(options), html_options)
end

def remote_function(options)
  javascript_options = options_for_ajax(options)
  update = ''
  if options[:update] && 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]
  url_options = url_options.merge(:escape => false) if 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

link_to_remote调用remote_function组织Prototype方法调用,然后调用link_to_function生成html标签和绑定onclick事件
看看例子:
<%= link_to_remote "Delete this post", :update => "posts", 
                                       :url => { :action => "destroy", :id => post.id } %>

<%= link_to_remote "Undo", :url => { :action => "undo", :n => word_counter },
                           :complete => "undoRequestCompleted(request)" %>

其中:update参数用来在Ajax异步调用完成后更新页面DOM元素,通常controller返回的结果为一个partial

2,periodically_call_remote用来做周期性url请求
def periodically_call_remote(options = {})
   frequency = options[:frequency] || 10
   code = "new PeriodicalExecuter(function() {#{remote_function(options)}}, #{frequency})"
   javascript_tag(code)
end

它new一个PeriodicalExecuter对象来完成周期性的工作,而callbacks等参数和link_to_remote一样,其中:frequency默认是10秒,看看例子:
<%= periodically_call_remote :update => 'time_div', 
                             :url => {:action => "time"},
                             :frequency => 1.0 %>


3,observe_form和observe_field用来监听form和field,如果内容改变,则触发function调用
def observe_form(form_id, options = {})
  if options[:frequency]
    build_observer('Form.Observer', form_id, options)
  else
    build_observer('Form.EventObserver', form_id, options)
  end
end

def observe_field(field_id, options = {})
  if options[:frequency] && options[:frequency] > 0
    build_observer('Form.Element.Observer', field_id, options)
  else
    build_observer('Form.Element.EventObserver', field_id, options)
  end
end

protected
def build_observer(klass, name, options = {})
  if options[:with] && !options[:with].include?("=")
    options[:with] = "'#{options[:with]}=' + value"
  else
    options[:with] ||= 'value' if options[:update]
  end
  callback = options[:function] || remote_function(options)
  javascript  = "new #{klass}('#{name}', "
  javascript << "#{options[:frequency]}, " if options[:frequency]
  javascript << "function(element, value) {"
  javascript << "#{callback}}"
  javascript << ", '#{options[: on]}'" if options[: on]
  javascript << ")"
  javascript_tag(javascript)
end

build_observer方法给出了生成JavaScript的实现(这段代码看着有点reflection的意思),然后使用javascript_tag来将JavaScript写入模板
看看例子:
<%= observe_form "entry-form", :frequency => 1,
                               :update => "live-preview",
                               :complete => "Element.show('live-preview')",
                               :url => {:action => "preview"} %>

<%= observe_field 'search',  :frequency => 0.5,
                             :update => 'target_id',
                             :url => { :controller => '<controller>', :action=> 'list' },
                             :with => "'search=' + escape(value)" %>


4,JavaScriptGenerator
      def update_page(&block)
        JavaScriptGenerator.new(@template, &block).to_s
      end

      class JavaScriptGenerator

        def initialize(context, &block) #:nodoc:
          @context, @lines = context, []
          include_helpers_from_context
          @context.instance_exec(self, &block)
        end

        private
          def include_helpers_from_context
            @context.extended_by.each do |mod|
              extend mod unless mod.name =~ /^ActionView::Helpers/
            end
            extend GeneratorMethods
          end

        module GeneratorMethods

          def to_s
            returning javascript = @lines * $/ do
              if ActionView::Base.debug_rjs
                source = javascript.dup
                javascript.replace "try {\n#{source}\n} catch (e) "
                javascript << "{ alert('RJS error:\\n\\n' + e.toString()); alert('#{source.gsub('\\','\0\0').gsub(/\r\n|\n|\r/, "\\n").gsub(/["']/) { |m| "\\#{m}" }}'); throw e }"
              end
            end
          end

          def [](id)
            JavaScriptElementProxy.new(self, id)
          end

          def select(pattern)
            JavaScriptElementCollectionProxy.new(self, pattern)
          end

          def insert_html(position, id, *options_for_render)
            insertion = position.to_s.camelize
            call "new Insertion.#{insertion}", id, render(*options_for_render)
          end

          def replace_html(id, *options_for_render)
            call 'Element.update', id, render(*options_for_render)
          end

          def replace(id, *options_for_render)
            call 'Element.replace', id, render(*options_for_render)
          end

          def remove(*ids)
            loop_on_multiple_args 'Element.remove', ids
          end

          def show(*ids)
            loop_on_multiple_args 'Element.show', ids
          end

          def hide(*ids)
            loop_on_multiple_args 'Element.hide', ids           
          end

          def toggle(*ids)
            loop_on_multiple_args 'Element.toggle', ids            
          end

          def alert(message)
            call 'alert', message
          end

          def redirect_to(location)
            assign 'window.location.href', @context.url_for(location)
          end

          def call(function, *arguments, &block)
            record "#{function}(#{arguments_for_call(arguments, block)})"
          end

          def delay(seconds = 1)
            record "setTimeout(function() {\n\n"
            yield
            record "}, #{(seconds * 1000).to_i})"
          end

          def visual_effect(name, id = nil, options = {})
            record @context.send(:visual_effect, name, id, options)
          end

          def sortable(id, options = {})
            record @context.send(:sortable_element_js, id, options)
          end

          def draggable(id, options = {})
            record @context.send(:draggable_element_js, id, options)
          end

          def drop_receiving(id, options = {})
            record @context.send(:drop_receiving_element_js, id, options)
          end

          def call(function, *arguments, &block)
            record "#{function}(#{arguments_for_call(arguments, block)})"
          end

          private

            def page
              self
            end

            def record(line)
              returning line = "#{line.to_s.chomp.gsub(/\;\z/, '')};" do
                self << line
              end
            end

            def method_missing(method, *arguments)
              JavaScriptProxy.new(self, method.to_s.camelize)
            end

        end
      end

我们终于看到了update_page这个helper方法了,它初始化JavaScriptGenerator对象,然后调用to_s方法返回生成的JavaScript代码
而update_page后的block中page方法返回self,即返回JavaScriptGenerator对象
JavaScriptGenerator类中定义了select,replace,hide,delay,visual_effect等我们熟悉的helper方法
而其中visual_effect,sortable,draggable和drop_receiving实际上会调用scriptaculous_helper中相应的方法
看看RJS的例子:
update_page do |page|
  page.insert_html :bottom, 'list', "<li>#{@item.name}</li>"
  page.visual_effect :highlight, 'list'
  page.hide 'status-indicator', 'cancel-link'
end

而[]方法和select方法分别使用JavaScriptElementProxy和JavaScriptElementCollectionProxy代理类

如果大家想了解和熟悉RJS语法,多看看javascript_helper.rb,prototype_helper.rb,scriptaculous_helper.rb

你可能感兴趣的:(JavaScript,html,Ajax,prototype,Rails)