0

所以在 Octopress 中,我希望有一个充满部分的文件夹,我可以很容易地在网站上以不同的方式显示。我想要一个链接到包含所有部分的文档中的部分标题列表。

这是我想使用的逻辑,但我不知道如何在 Octopress 中实际实现它。

class Collection
  attr_accessor :directory, :files

  def new(dir)
    self.files = []
    self.directory = dir
    load_files
  end

  def list(options={})
    # Handle options here
    files
  end

private

  def load_files
    files_in(@directory).each do |file| # Psuedo-code
      f = Jekyll::FileObject.new(file.read) # Also fictional, but I imagine something similar exists
      f.text # Would return the actual content
      f.yaml # Would return a hash of options from the YAML front matter

      files << f
    end
  end
end

然后我想加载部分集合:

api_methods = Collection.new("api_methods").list(:alphabetical => true, :method_type => "public")

在液体中,列出目录:

<ul>
{% for partial in api_methods %}
  <li><a href="{% partial.yaml.url %}">{% partial.yaml.name %}</a></li>
{% endfor %}
</ul>

在液体中,列出全文:

{% for partial in api_methods %}
  <h2>{% partial.yaml.name %}</h2>

  {% partial.text %}
{% endfor %}

因此,这应该有望让您对我正在尝试做的事情有一个基本的了解,但我不知道如何实际实现这一点。

4

1 回答 1

1

好的,这就是我完成我打算做的事情的方式。

首先,我必须认真阅读 Jekyll 的源代码。一旦我对它的工作原理有了基本的了解,我就可以进去修补某些东西。

基本上,Liquid 是一个非常简单的模板系统,您不能使用它转换任何数据。您必须以“有效负载”的形式将数据发送到 Liquid,这只是一个大哈希。因此,如果我想转换一些数据,我必须在将其发送到 Liquid 之前在 Ruby 中进行。

所以我做了一个东西叫PayloadExtension

插件/site.rb

module Jekyll
  class Site
    alias :old_site_payload :site_payload

    def site_payload
      PayloadExtension.payload(old_site_payload)
    end
  end
end

插件/payload_extension.rb

module Jekyll
  class PayloadExtension
    extend OctopressFilters

    def self.payload(payload)
      pages = payload['site']['pages']

      # Filter pages by Public API methods category and sort by title

      public_api_methods = category_filter(pages, "Public API methods")
      public_api_methods.sort! {|a, b| a.data['title'] <=> b.data['title'] }
      public_api_methods = pre_render(public_api_methods)

      payload['site']['public_api_methods'] = public_api_methods

      payload
    end

  private

    def self.category_filter(pages, category)
      pages.reject do |page|
        cat = page.data['category']

        !cat || cat.empty? || cat != category || ( cat.is_a?(Array) && !cat.include?(category) )
      end
    end

    def self.pre_render(pages)
      pages.collect do |page|
        content = pre_filter(page.content)
        page.data['rendered'] = Liquid::Template.parse(content).render
        page
      end
    end

  end
end

不幸的是,我无法搞砸Convertible#do_layout,所以我在这个阶段可以访问的数据不是最终的东西,但这没关系,因为它很容易解决。

我还必须自己手动渲染页面的内容,因为只是 put{{ page.content }}会吐出尚未完全解析的文本。

现在在实际的 Liquid 模板中,我可以这样做:

{% for page in site.public_api_methods %}
  <h2>{{ page.title }} / {{ page.http_method | upcase }}</h2>
  {{ page.rendered }}
{% endfor %}

<h3>Public methods</h3>
<ul>
  {% for page in site.public_api_methods %}
    <li><a href="/api/public-methods/#{{ page.title | slugify }}-{{ page.http_method }}">{{ page.title }}</a></li>
  {% endfor %}
</ul>
于 2012-04-20T21:57:05.393 回答