Sha256: ca42f86732c33613ff5847e1f4aa5c88eb16a2d0f0caca4a6cff856cc3dcc68d

Contents?: true

Size: 1.92 KB

Versions: 4

Compression:

Stored size: 1.92 KB

Contents

module HTML
  class Pipeline

    # HTML filter that adds a 'name' attribute to all headers
    # in a document, so they can be accessed from a table of contents
    #
    # Context options:
    #   :toc_minimal_length (required) - Only add the table of contents to text with this number of characters
    #   :toc_header (required) - Introduce the table of contents with this header
    #
    class TableOfContentsFilter < Filter

      def call
        headers = Hash.new 0
        was = 1
        toc = ""
        doc.css('h2, h3, h4, h5, h6').each do |node|
          level = node.name.scan(/\d/).first.to_i
          name = node.text.downcase
          name.gsub!(/[^\w\- ]/, '') # remove punctuation
          name.gsub!(' ', '-') # replace spaces with dash
          name = EscapeUtils.escape_uri(name) # escape extended UTF-8 chars

          uniq = (headers[name] > 0) ? "-#{headers[name]}" : ''
          headers[name] += 1
          node['id'] = "#{name}#{uniq}"

          if was < level
            while was < level
              toc << "<ul>\n<li>"
              was += 1
            end
          else
            toc << "</li>\n"
            while was > level
              toc << "</ul></li>\n"
              was -= 1
            end
            toc << "<li>"
          end
          toc << "<a href=\"##{name}#{uniq}\">#{node.inner_html}</a>"
        end

        length = 0
        doc.traverse {|node| length += node.text.length if node.text? }
        return doc unless length >= context[:toc_minimal_length]

        while was > 1
          toc << "</li>\n</ul>\n"
          was -= 1
        end
        toc.sub!('<ul>', '<ul class="toc">')

        unless headers.empty?
          first_child = doc.child
          first_child.add_previous_sibling context[:toc_header]
          first_child.add_previous_sibling toc
        end
        doc
      end

      def validate
        needs :toc_minimal_length, :toc_header
      end
    end

  end
end

Version data entries

4 entries across 4 versions & 1 rubygems

Version Path
html-pipeline-linuxfr-0.14.5 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.4 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.3 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.2 lib/html/pipeline/toc_filter.rb