Rails: метод захвата

Следующий код из Ryan Bates RailsCasts, в котором он превращает главную страницу блога в календарь, так что статьи отображаются как ссылки на дни. Следующий вспомогательный модуль создает Календарь. У меня есть два вопроса об этом коде

  • В методе day_cell он использует метод под названием capture. Я нашел некоторые документы, но я все еще не могу понять, как работает захват в этом контексте. Кроме того, что такое &callback, переданное как аргумент для захвата? Это будет тот же самый :callback, который передается Struct.new? Если да, то как он попадает в захват? Что такое: обратный вызов, переданный в Struct?

    def day_cell(day)
      content_tag :td, view.capture(day, &callback), class: day_classes(day)
    end
    

исходный код

module CalendarHelper
  def calendar(date = Date.today, &block)
    binding.pry
    Calendar.new(self, date, block).table

  end

  class Calendar < Struct.new(:view, :date, :callback)
    HEADER = %w[Sunday Monday Tuesday Wednesday Thursday Friday Saturday]
    START_DAY = :sunday

    delegate :content_tag, to: :view

    def table
      content_tag :table, class: "calendar" do
        header + week_rows
      end
    end

    def header
      content_tag :tr do
        HEADER.map { |day| content_tag :th, day }.join.html_safe
      end
    end

    def week_rows
      weeks.map do |week|
        content_tag :tr do
          week.map { |day| day_cell(day) }.join.html_safe
        end
      end.join.html_safe
    end

    def day_cell(day)
      content_tag :td, view.capture(day, &callback), class: day_classes(day)
    end

    def day_classes(day)
      classes = []
      classes << "today" if day == Date.today
      classes << "notmonth" if day.month != date.month
      classes.empty? ? nil : classes.join(" ")
    end

    def weeks
      first = date.beginning_of_month.beginning_of_week(START_DAY)
      last = date.end_of_month.end_of_week(START_DAY)
      (first..last).to_a.in_groups_of(7)
    end
  end
end

Ответ 1

Я провел свое исследование, и я, наконец, разгадал тайну.

Итак, несколько вещей для начала; как обычно документация не очень понятна, метод capture (* args) должен захватить кусок шаблона в переменную, но он не углубляется в объяснение того, что вы можете передавать переменные в захваченный кусок шаблона, это, конечно, происходит в виде блока

исходный код Райан Бэйт Экран календаря:

<div id="articles">
  <h2 id="month">
    <%= link_to "<", date: @date.prev_month %>
    <%= @date.strftime("%B %Y") %>
    <%= link_to ">", date: @date.next_month %>
  </h2>
  <%= calendar @date do |date| %>
    <%= date.day %>
    <% if @articles_by_date[date] %>
      <ul>
        <% @articles_by_date[date].each do |article| %>
          <li><%= link_to article.name, article %></li>
        <% end %>
      </ul>
    <% end %>
  <% end %>
</div>

В приведенном выше коде блок будет исключительно этой частью:

do |date| %>
  <%= date.day %>
  <% if @articles_by_date[date] %>
    <ul>
      <% @articles_by_date[date].each do |article| %>
        <li><%= link_to article.name, article %></li>
      <% end %>
    </ul>
  <% end %>
<% end %>

Итак, когда он делает этот вызов:

content_tag :td, view.capture(day, &callback), class: day_classes(day)

в частности:

view.capture(day, &callback)

Что здесь происходит, так это то, что он передал аргумент day для блока выше как параметр | date | (в блоке).

Что нужно понимать здесь, так это то, что в контексте проблемы (создание 30-дневного календаря); каждый день месяца передается методу захвата вместе с частью шаблона (& callback), делая это.. в результате отображает блок выше для каждого дня данного месяца. Конечным этапом, конечно же, является.. Размещение этого отображаемого содержимого (для каждого дня) в качестве содержимого для content_tag: td

Последняя заметка; Райан вызывает метод захвата в переменной view, он также не указан в документации, но он упоминает во время ScreenCast, что ему нужно это представление в качестве "прокси" для доступа к представлению, и, конечно, представление является единственным, у которого есть доступ к методам ViewHelper.

Итак, в целом, это очень красивый код, но он только красив, как только вы понимаете, что он делает. Итак, я согласен, что с первого взгляда очень запутанно.

Надеюсь, это поможет, это лучшее объяснение, которое я мог бы придумать.:)