Class: Premailer
- Inherits:
-
Object
- Object
- Premailer
- Includes:
- CssParser, HtmlToPlainText, Warnings
- Defined in:
- lib/premailer/adapter.rb,
lib/premailer/version.rb,
lib/premailer/premailer.rb,
lib/premailer/adapter/nokogiri.rb,
lib/premailer/adapter/nokogumbo.rb,
lib/premailer/adapter/nokogiri_fast.rb
Defined Under Namespace
Constant Summary
- VERSION =
Premailer version.
'1.11.0'.freeze
- CLIENT_SUPPORT_FILE =
File.dirname(__FILE__) + '/../../misc/client_support.yaml'
- RE_UNMERGABLE_SELECTORS =
Unmergable selectors regexp.
/(\:(visited|active|hover|focus|after|before|selection|target|first\-(line|letter))|^\@)/i
- RE_RESET_SELECTORS =
Reset selectors regexp.
/^(\:\#outlook|body.*|\.ReadMsgBody|\.ExternalClass|img|\#backgroundTable)$/
- HTML_ENTITIES =
list of HTMLEntities to fix source: http://stackoverflow.com/questions/2812781/how-to-convert-webpage-apostrophe-8217-to-ascii-39-in-ruby-1-
{ "’" => "'", "…" => "...", "‘" => "'", "‚" => ',', "‛" => "'", "“" => '"', "”" => '"', "‐" => '-', "–" => '-', "—" => '--', "―" => '--' }
- RELATED_ATTRIBUTES =
TODO:
too much repetition
TODO:background=""
list of CSS attributes that can be rendered as HTML attributes
{ 'h1' => {'text-align' => 'align'}, 'h2' => {'text-align' => 'align'}, 'h3' => {'text-align' => 'align'}, 'h4' => {'text-align' => 'align'}, 'h5' => {'text-align' => 'align'}, 'h6' => {'text-align' => 'align'}, 'p' => {'text-align' => 'align'}, 'div' => {'text-align' => 'align'}, 'blockquote' => {'text-align' => 'align'}, 'body' => {'background-color' => 'bgcolor'}, 'table' => { '-premailer-align' => 'align', 'background-color' => 'bgcolor', 'background-image' => 'background', '-premailer-width' => 'width', '-premailer-height' => 'height', '-premailer-cellpadding' => 'cellpadding', '-premailer-cellspacing' => 'cellspacing' }, 'tr' => { 'text-align' => 'align', 'background-color' => 'bgcolor', '-premailer-height' => 'height' }, 'th' => { 'text-align' => 'align', 'background-color' => 'bgcolor', 'vertical-align' => 'valign', '-premailer-width' => 'width', '-premailer-height' => 'height' }, 'td' => { 'text-align' => 'align', 'background-color' => 'bgcolor', 'vertical-align' => 'valign', '-premailer-width' => 'width', '-premailer-height' => 'height' }, 'img' => { 'float' => 'align', '-premailer-width' => 'width', '-premailer-height' => 'height' } }
- WARN_LABEL =
Waning level names
%w(NONE SAFE POOR RISKY)
Constants included from Warnings
Warnings::NONE, Warnings::POOR, Warnings::RISKY, Warnings::SAFE
Instance Attribute Summary collapse
-
#base_dir ⇒ String
readonly
base directory used to resolve links for local files.
-
#base_url ⇒ Object
readonly
base URL used to resolve links.
-
#doc ⇒ Object
readonly
source HTML document (Nokogiri/Nokogumbo).
-
#html_file ⇒ Object
readonly
URI of the HTML file used.
-
#processed_doc ⇒ Object
readonly
processed HTML document (Nokogiri/Nokogumbo).
-
#unmergable_rules ⇒ Object
readonly
unmergeable CSS rules to be preserved in the head (CssParser).
Class Method Summary collapse
- .canonicalize(uri) ⇒ Object
-
.local_data?(data) ⇒ Boolean
Test the passed variable to see if we are in local or remote mode.
Instance Method Summary collapse
- #append_query_string(doc, qs) ⇒ Object
-
#check_client_support ⇒ Object
Check CLIENT_SUPPORT_FILE for any CSS warnings.
-
#convert_inline_links(doc, base_uri) ⇒ Object
Convert relative links to absolute links.
-
#initialize(html, options = {}) ⇒ Premailer
constructor
Create a new Premailer object.
-
#is_xhtml? ⇒ Boolean
Check for an XHTML doctype.
-
#warnings ⇒ Array(Hash)
CSS warnings.
Methods included from HtmlToPlainText
Constructor Details
#initialize(html, options = {}) ⇒ Premailer
Create a new Premailer object.
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 |
# File 'lib/premailer/premailer.rb', line 181 def initialize(html, = {}) @options = {:warn_level => Warnings::SAFE, :line_length => 65, :link_query_string => nil, :base_url => nil, :rgb_to_hex_attributes => true, :remove_classes => false, :remove_ids => false, :remove_comments => false, :remove_scripts => true, :reset_contenteditable => true, :css => [], :css_to_attributes => true, :preserve_style_attribute => false, :with_html_string => false, :css_string => nil, :preserve_styles => false, :preserve_reset => true, :verbose => false, :debug => false, :io_exceptions => false, :include_link_tags => true, :include_style_tags => true, :input_encoding => 'ASCII-8BIT', :output_encoding => nil, :replace_html_entities => false, :escape_url_attributes => true, :unescaped_ampersand => false, :create_shorthands => true, :html_fragment => false, :adapter => Adapter.use, }.merge() @html_file = html @is_local_file = @options[:with_html_string] || Premailer.local_data?(html) @css_files = [@options[:css]].flatten @css_warnings = [] @base_url = nil @base_dir = nil @unmergable_rules = nil if @options[:base_url] @base_url = Addressable::URI.parse(@options.delete(:base_url)) elsif not @is_local_file @base_url = Addressable::URI.parse(@html_file) end @css_parser = CssParser::Parser.new({ :absolute_paths => true, :import => true, :io_exceptions => @options[:io_exceptions] }) @adapter_class = Adapter.find @options[:adapter] self.class.send(:include, @adapter_class) @doc = load_html(@html_file) @processed_doc = @doc @processed_doc = convert_inline_links(@processed_doc, @base_url) if @base_url if [:link_query_string] @processed_doc = append_query_string(@processed_doc, [:link_query_string]) end load_css_from_html! end |
Instance Attribute Details
#base_dir ⇒ String (readonly)
base directory used to resolve links for local files
119 120 121 |
# File 'lib/premailer/premailer.rb', line 119 def base_dir @base_dir end |
#base_url ⇒ Object (readonly)
base URL used to resolve links
115 116 117 |
# File 'lib/premailer/premailer.rb', line 115 def base_url @base_url end |
#doc ⇒ Object (readonly)
source HTML document (Nokogiri/Nokogumbo)
128 129 130 |
# File 'lib/premailer/premailer.rb', line 128 def doc @doc end |
#html_file ⇒ Object (readonly)
URI of the HTML file used
112 113 114 |
# File 'lib/premailer/premailer.rb', line 112 def html_file @html_file end |
#processed_doc ⇒ Object (readonly)
processed HTML document (Nokogiri/Nokogumbo)
125 126 127 |
# File 'lib/premailer/premailer.rb', line 125 def processed_doc @processed_doc end |
#unmergable_rules ⇒ Object (readonly)
unmergeable CSS rules to be preserved in the head (CssParser)
122 123 124 |
# File 'lib/premailer/premailer.rb', line 122 def unmergable_rules @unmergable_rules end |
Class Method Details
.canonicalize(uri) ⇒ Object
497 498 499 500 501 502 503 504 505 506 507 |
# File 'lib/premailer/premailer.rb', line 497 def self.canonicalize(uri) # :nodoc: u = uri.kind_of?(Addressable::URI) ? uri : Addressable::URI.parse(uri.to_s) u.normalize! newpath = u.path while newpath.gsub!(%r{([^/]+)/\.\./?}) { |match| $1 == '..' ? match : '' } do end newpath = newpath.gsub(%r{/\./}, '/').sub(%r{/\.\z}, '/') u.path = newpath u.to_s end |
.local_data?(data) ⇒ Boolean
Test the passed variable to see if we are in local or remote mode.
IO objects return true, as do strings that look like URLs.
491 492 493 494 |
# File 'lib/premailer/premailer.rb', line 491 def self.local_data?(data) return false if data.kind_of?(String) && data =~ /\A(?:(https?|ftp):)\/\//i true end |
Instance Method Details
#append_query_string(doc, qs) ⇒ Object
353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 |
# File 'lib/premailer/premailer.rb', line 353 def append_query_string(doc, qs) return doc if qs.nil? qs.to_s.gsub!(/^[\?]*/, '').strip! return doc if qs.empty? begin current_host = @base_url.host rescue current_host = nil end $stderr.puts "Attempting to append_query_string: #{qs}" if @options[:verbose] doc.search('a').each do|el| href = el.attributes['href'].to_s.strip next if href.nil? or href.empty? next if href[0,1] =~ /[\#\{\[\<\%]/ # don't bother with anchors or special-looking links begin href = Addressable::URI.parse(href) if current_host and href.host != nil and href.host != current_host $stderr.puts "Skipping append_query_string for: #{href.to_s} because host is no good" if @options[:verbose] next end if href.scheme and href.scheme != 'http' and href.scheme != 'https' puts "Skipping append_query_string for: #{href.to_s} because scheme is no good" if @options[:verbose] next end if href.query and not href.query.empty? amp = @options[:unescaped_ampersand] ? '&' : '&' href.query = href.query + amp + qs else href.query = qs end el['href'] = href.to_s rescue Addressable::URI::InvalidURIError => e $stderr.puts "Skipping append_query_string for: #{href.to_s} (#{e.})" if @options[:verbose] next end end doc end |
#check_client_support ⇒ Object
Check CLIENT_SUPPORT_FILE for any CSS warnings
510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 |
# File 'lib/premailer/premailer.rb', line 510 def check_client_support # :nodoc: @client_support ||= YAML::load(File.open(CLIENT_SUPPORT_FILE)) warnings = [] properties = [] # Get a list off CSS properties @processed_doc.search("*[@style]").each do |el| style_url = el.attributes['style'].to_s.gsub(/([\w\-]+)[\s]*\:/i) do |s| properties.push($1) end end properties.uniq! property_support = @client_support['css_properties'] properties.each do |prop| if property_support.include?(prop) and property_support[prop].include?('support') and property_support[prop]['support'] >= @options[:warn_level] warnings.push({:message => "#{prop} CSS property", :level => WARN_LABEL[property_support[prop]['support']], :clients => property_support[prop]['unsupported_in'].join(', ')}) end end @client_support['attributes'].each do |attribute, data| next unless data['support'] >= @options[:warn_level] if @doc.search("*[@#{attribute}]").length > 0 warnings.push({:message => "#{attribute} HTML attribute", :level => WARN_LABEL[data['support']], :clients => data['unsupported_in'].join(', ')}) end end @client_support['elements'].each do |element, data| next unless data['support'] >= @options[:warn_level] if @doc.search(element).length > 0 warnings.push({:message => "#{element} HTML element", :level => WARN_LABEL[data['support']], :clients => data['unsupported_in'].join(', ')}) end end warnings end |
#convert_inline_links(doc, base_uri) ⇒ Object
Convert relative links to absolute links.
Processes href src and background attributes as well as CSS url() declarations found in inline style attributes.
doc is a document and base_uri is either a string or a URI.
Returns a document.
419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 |
# File 'lib/premailer/premailer.rb', line 419 def convert_inline_links(doc, base_uri) # :nodoc: base_uri = Addressable::URI.parse(base_uri) unless base_uri.kind_of?(Addressable::URI) append_qs = @options[:link_query_string] || '' escape_attrs = @options[:escape_url_attributes] ['href', 'src', 'background'].each do |attribute| = doc.search("*[@#{attribute}]") next if .empty? .each do |tag| # skip links that look like they have merge tags # and mailto, ftp, etc... if tag.attributes[attribute].to_s =~ /^([\%\<\{\#\[]|data:|tel:|file:|sms:|callto:|facetime:|mailto:|ftp:|gopher:|cid:)/i next end if tag.attributes[attribute].to_s =~ /^http/i begin merged = Addressable::URI.parse(tag.attributes[attribute]) rescue; next; end else begin merged = Premailer.resolve_link(tag.attributes[attribute].to_s, base_uri) rescue begin next unless escape_attrs merged = Premailer.resolve_link(Addressable::URI.escape(tag.attributes[attribute].to_s), base_uri) rescue; end end end # make sure 'merged' is a URI merged = Addressable::URI.parse(merged.to_s) unless merged.kind_of?(Addressable::URI) tag[attribute] = merged.to_s end # end of each tag end # end of each attrs doc.search("*[@style]").each do |el| el['style'] = CssParser.convert_uris(el.attributes['style'].to_s, base_uri) end doc end |
#is_xhtml? ⇒ Boolean
Check for an XHTML doctype
404 405 406 407 408 409 |
# File 'lib/premailer/premailer.rb', line 404 def is_xhtml? intro = @doc.to_html.strip.split("\n")[0..2].join(' ') is_xhtml = !!(intro =~ /w3c\/\/[\s]*dtd[\s]+xhtml/i) $stderr.puts "Is XHTML? #{is_xhtml.inspect}\nChecked:\n#{intro}" if @options[:debug] is_xhtml end |