Sha256: ff93d118ec4e0eed55e9f689da2847b83fcdb9ae0dd828090347952b1fa74f54

Contents?: true

Size: 1.86 KB

Versions: 30

Compression:

Stored size: 1.86 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!(/[^\p{Word}\- ]/u, '') # remove punctuation
          name.gsub!(' ', '-') # replace spaces with dash

          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

30 entries across 30 versions & 1 rubygems

Version Path
html-pipeline-linuxfr-0.15.4 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.15.3 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.30 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.15.2 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.15.1 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.29 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.15.0 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.28 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.27 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.26 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.25 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.24 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.23 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.22 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.21 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.20 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.19 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.18 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.17 lib/html/pipeline/toc_filter.rb
html-pipeline-linuxfr-0.14.16 lib/html/pipeline/toc_filter.rb