# frozen_string_literal: true module Asciidoctor class SyntaxHighlighter::RougeAdapter < SyntaxHighlighter::Base register_for 'rouge' def initialize *args super @requires_stylesheet = nil @style = nil end def highlight? library_available? end def highlight node, source, lang, opts lexer = (::Rouge::Lexer.find_fancy lang) || ::Rouge::Lexers::PlainText lexer_opts = lexer.tag == 'php' && !(node.option? 'mixed') ? { start_inline: true } : {} @style ||= (style = opts[:style]) && (style_available? style) || DEFAULT_STYLE if opts[:css_mode] == :class @requires_stylesheet = true formatter = ::Rouge::Formatters::HTML.new @style else formatter = ::Rouge::Formatters::HTMLInline.new @style end if (highlight_lines = opts[:highlight_lines]) formatter = RougeExt::Formatters::HTMLLineHighlighter.new formatter, lines: highlight_lines end if opts[:number_lines] formatter = RougeExt::Formatters::HTMLTable.new formatter, start_line: opts[:start_line_number] if opts[:callouts] return [(highlighted = formatter.format lexer.lex source, lexer_opts), (idx = highlighted.index CodeCellStartTagCs) ? idx + CodeCellStartTagCs.length : nil] end end formatter.format lexer.lex source, lexer_opts end def format node, lang, opts if (query_idx = lang.index '?') lang = lang.slice 0, query_idx end if opts[:css_mode] != :class && (@style = (style = opts[:style]) && (style_available? style) || DEFAULT_STYLE) && (pre_style_attr_val = base_style @style) opts[:transform] = proc {|pre| pre['style'] = pre_style_attr_val } end super end def docinfo? location @requires_stylesheet && location == :footer end def docinfo location, doc, opts if opts[:linkcss] %() else %() end end def write_stylesheet? doc @requires_stylesheet end def write_stylesheet doc, to_dir ::File.write (::File.join to_dir, (stylesheet_basename @style)), (read_stylesheet @style), mode: FILE_WRITE_MODE end module Loader private def library_available? (@@library_status ||= load_library) == :loaded ? true : nil end def load_library (defined? RougeExt) ? :loaded : (Helpers.require_library %(#{::File.dirname __dir__}/rouge_ext), 'rouge', :warn).nil? ? :unavailable : :loaded end end module Styles include Loader def read_stylesheet style library_available? ? @@stylesheet_cache[style || DEFAULT_STYLE] : '/* Rouge CSS disabled because Rouge is not available. */' end def stylesheet_basename style %(rouge-#{style || DEFAULT_STYLE}.css) end private def base_style style library_available? ? @@base_style_cache[style || DEFAULT_STYLE] : nil end def style_available? style (::Rouge::Theme.find style) && style end @@base_style_cache = ::Hash.new do |cache, key| base_style = (theme = ::Rouge::Theme.find key).base_style (val = base_style[:fg]) && ((style ||= []) << %(color: #{theme.palette val})) (val = base_style[:bg]) && ((style ||= []) << %(background-color: #{theme.palette val})) @@base_style_cache = cache.merge key => (resolved_base_style = style && (style.join ';')) resolved_base_style end @@stylesheet_cache = ::Hash.new do |cache, key| @@stylesheet_cache = cache.merge key => (stylesheet = ((::Rouge::Theme.find key).render scope: BASE_SELECTOR)) stylesheet end DEFAULT_STYLE = 'github' BASE_SELECTOR = 'pre.rouge' private_constant :BASE_SELECTOR end extend Styles # exports static methods include Loader, Styles # adds methods to instance CodeCellStartTagCs = '