| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
 | require 'action_view/renderer/abstract_renderer'
module ActionView
  class PartialRenderer < AbstractRenderer #:nodoc:
    PARTIAL_NAMES = Hash.new {|h,k| h[k] = {} }
    def initialize(view)
      super
      @partial_names = PARTIAL_NAMES[@view.controller.class.name]
    end
    def setup(options, block)
      partial = options[:partial]
      @options = options
      @locals  = options[:locals] || {}
      @block   = block
      if String === partial
        @object     = options[:object]
        @path       = partial
        @collection = collection
      else
        @object = partial
        if @collection = collection_from_object || collection
          paths = @collection_data = @collection.map { |o| partial_path(o) }
          @path = paths.uniq.size == 1 ? paths.first : nil
        else
          @path = partial_path
        end
      end
      if @path
        @variable, @variable_counter = retrieve_variable(@path)
      else
        paths.map! { |path| retrieve_variable(path).unshift(path) }
      end
      self
    end
    def render
      wrap_formats(@path) do
        identifier = ((@template = find_partial) ? @template.identifier : @path)
        if @collection
          instrument(:collection, :identifier => identifier || "collection", :count => @collection.size) do
            render_collection
          end
        else
          instrument(:partial, :identifier => identifier) do
            render_partial
          end
        end
      end
    end
    def render_collection
      return nil if @collection.blank?
      if @options.key?(:spacer_template)
        spacer = find_template(@options[:spacer_template]).render(@view, @locals)
      end
      result = @template ? collection_with_template : collection_without_template
      result.join(spacer).html_safe
    end
    def render_partial
      locals, view, block = @locals, @view, @block
      object, as = @object, @variable
      if !block && (layout = @options[:layout])
        layout = find_template(layout)
      end
      object ||= locals[as]
      locals[as] = object
      content = @template.render(view, locals) do |*name|
        view._layout_for(*name, &block)
      end
      content = layout.render(view, locals){ content } if layout
      content
    end
    private
    def collection
      if @options.key?(:collection)
        collection = @options[:collection]
        collection.respond_to?(:to_ary) ? collection.to_ary : []
      end
    end
    def collection_from_object
      if @object.respond_to?(:to_ary)
        @object.to_ary
      end
    end
    def find_partial
      if path = @path
        locals = @locals.keys
        locals << @variable
        locals << @variable_counter if @collection
        find_template(path, locals)
      end
    end 
    def find_template(path=@path, locals=@locals.keys)
      prefix = @view.controller_prefix unless path.include?(?/)
      @lookup_context.find_template(path, prefix, true, locals)
    end
    def collection_with_template
      segments, locals, template = [], @locals, @template
      as, counter = @variable, @variable_counter
      locals[counter] = -1
      @collection.each do |object|
        locals[counter] += 1
        locals[as] = object
        segments << template.render(@view, locals)
      end
      segments
    end
    def collection_without_template
      segments, locals, collection_data = [], @locals, @collection_data
      index, template, cache = -1, nil, {}
      keys = @locals.keys
      @collection.each_with_index do |object, i|
        path, *data = collection_data[i]
        template = (cache[path] ||= find_template(path, keys + data))
        locals[data[0]] = object
        locals[data[1]] = (index += 1)
        segments << template.render(@view, locals)
      end
      @template = template
      segments
    end
    def partial_path(object = @object)
      @partial_names[object.class.name] ||= begin
        object = object.to_model if object.respond_to?(:to_model)
        object.class.model_name.partial_path.dup.tap do |partial|
          path = @view.controller_prefix
          partial.insert(0, "#{File.dirname(path)}/") if partial.include?(?/) && path.include?(?/)
        end
      end
    end
    def retrieve_variable(path)
      variable = @options[:as] || path[%r'_?(\w+)(\.\w+)*$', 1].to_sym
      variable_counter = :"#{variable}_counter" if @collection
      [variable, variable_counter]
    end
  end
end
 |