diff --git a/doc-src/Gemfile b/doc-src/Gemfile index 5eebcc3e..5ad23a24 100644 --- a/doc-src/Gemfile +++ b/doc-src/Gemfile @@ -16,6 +16,6 @@ gem 'compass', :path => ".." gem 'compass-susy-plugin', ">=0.7.0.pre8" gem 'css-slideshow', "0.2.0" gem 'json' -gem 'css_parser' +gem 'css_parser', "1.0.1" gem 'ruby-prof' diff --git a/doc-src/Gemfile.lock b/doc-src/Gemfile.lock index c2090be9..6af6a34d 100644 --- a/doc-src/Gemfile.lock +++ b/doc-src/Gemfile.lock @@ -23,7 +23,7 @@ GEM cri (1.0.1) css-slideshow (0.2.0) compass (>= 0.10.0.rc3) - css_parser (1.1.9) + css_parser (1.0.1) fssm (0.1.2) haml (3.1.0.alpha.147) i18n (0.4.2) @@ -51,7 +51,7 @@ DEPENDENCIES compass! compass-susy-plugin (>= 0.7.0.pre8) css-slideshow (= 0.2.0) - css_parser + css_parser (= 1.0.1) fssm (= 0.1.2) haml (>= 3.1.0.alpha.36) json diff --git a/lib/compass/sass_extensions/functions/constants.rb b/lib/compass/sass_extensions/functions/constants.rb index 846787f6..02be8ef8 100644 --- a/lib/compass/sass_extensions/functions/constants.rb +++ b/lib/compass/sass_extensions/functions/constants.rb @@ -1,5 +1,12 @@ module Compass::SassExtensions::Functions::Constants if defined?(Sass::Script::List) + POSITIONS = /top|bottom|left|right|center/ + def is_position(position) + Sass::Script::Bool.new(position.is_a?(Sass::Script::String) && !!(position.value =~ POSITIONS)) + end + def is_position_list(position_list) + Sass::Script::Bool.new(position_list.is_a?(Sass::Script::List) && position_list.value.all?{|p| is_position(p).to_bool}) + end # returns the opposite position of a side or corner. def opposite_position(position) position = unless position.is_a?(Sass::Script::List) diff --git a/lib/compass/sass_extensions/functions/gradient_support.rb b/lib/compass/sass_extensions/functions/gradient_support.rb index a1f466e9..2c2bc207 100644 --- a/lib/compass/sass_extensions/functions/gradient_support.rb +++ b/lib/compass/sass_extensions/functions/gradient_support.rb @@ -34,15 +34,15 @@ module Compass::SassExtensions::Functions::GradientSupport end class RadialGradient < Sass::Script::Literal - attr_accessor :position_and_angle, :shape_and_size, :color_stops + attr_accessor :position_or_angle, :shape_and_size, :color_stops def children - [color_stops, position_and_angle, shape_and_size].compact + [color_stops, position_or_angle, shape_and_size].compact end - def initialize(position_and_angle, shape_and_size, color_stops) + def initialize(position_or_angle, shape_and_size, color_stops) unless color_stops.value.size >= 2 raise Sass::SyntaxError, "At least two color stops are required for a radial-gradient" end - self.position_and_angle = position_and_angle + self.position_or_angle = position_or_angle self.shape_and_size = shape_and_size self.color_stops = color_stops end @@ -51,22 +51,31 @@ module Compass::SassExtensions::Functions::GradientSupport end def to_s(options = self.options) s = "radial-gradient(" - s << position_and_angle.to_s(options) << ", " if position_and_angle + s << position_or_angle.to_s(options) << ", " if position_or_angle s << shape_and_size.to_s(options) << ", " if shape_and_size s << color_stops.to_s(options) s << ")" end def supports?(aspect) - GRADIENT_ASPECTS.include?(aspect) + if aspect == "svg" + if position_or_angle.nil? || is_position(position_or_angle).to_bool || is_position_list(position_or_angle).to_bool + true + else + Compass::Util.compass_warn("Warning: Angle-based gradients are not yet supported in SVG. Found: #{position_or_angle}") + false + end + else + GRADIENT_ASPECTS.include?(aspect) + end end def has_aspect? true end def to_webkit(options = self.options) args = [ - grad_point(position_and_angle || _center_position), + grad_point(position_or_angle || _center_position), Sass::Script::String.new("0"), - grad_point(position_and_angle || _center_position), + grad_point(position_or_angle || _center_position), grad_end_position(color_stops, Sass::Script::Bool.new(true)), grad_color_stops(color_stops) ] @@ -82,7 +91,7 @@ module Compass::SassExtensions::Functions::GradientSupport end def to_svg(options = self.options) # XXX Add shape support if possible - radial_svg_gradient(color_stops, position_and_angle || _center_position) + radial_svg_gradient(color_stops, position_or_angle || _center_position) end def to_pie(options = self.options) Compass::Logger.new.record(:warning, "PIE does not support radial-gradient.") @@ -94,15 +103,15 @@ module Compass::SassExtensions::Functions::GradientSupport end class LinearGradient < Sass::Script::Literal - attr_accessor :color_stops, :position_and_angle + attr_accessor :color_stops, :position_or_angle def children - [color_stops, position_and_angle].compact + [color_stops, position_or_angle].compact end - def initialize(position_and_angle, color_stops) + def initialize(position_or_angle, color_stops) unless color_stops.value.size >= 2 raise Sass::SyntaxError, "At least two color stops are required for a linear-gradient" end - self.position_and_angle = position_and_angle + self.position_or_angle = position_or_angle self.color_stops = color_stops end def inspect @@ -110,20 +119,29 @@ module Compass::SassExtensions::Functions::GradientSupport end def to_s(options = self.options) s = "linear-gradient(" - s << position_and_angle.to_s(options) << ", " if position_and_angle + s << position_or_angle.to_s(options) << ", " if position_or_angle s << color_stops.to_s(options) s << ")" end def supports?(aspect) - GRADIENT_ASPECTS.include?(aspect) + if aspect == "svg" + if position_or_angle.nil? || is_position(position_or_angle).to_bool || is_position_list(position_or_angle).to_bool + true + else + Compass::Util.compass_warn("Warning: Angle-based gradients are not yet supported in SVG. Found: #{position_or_angle}") + false + end + else + GRADIENT_ASPECTS.include?(aspect) + end end def has_aspect? true end def to_webkit(options = self.options) args = [] - args << grad_point(position_and_angle || Sass::Script::String.new("top")) - args << grad_point(opposite_position(position_and_angle || Sass::Script::String.new("top"))) + args << grad_point(position_or_angle || Sass::Script::String.new("top")) + args << grad_point(opposite_position(position_or_angle || Sass::Script::String.new("top"))) args << grad_color_stops(color_stops) args.each{|a| a.options = options} Sass::Script::String.new("-webkit-gradient(linear, #{args.join(', ')})") @@ -135,7 +153,7 @@ module Compass::SassExtensions::Functions::GradientSupport Sass::Script::String.new("-o-#{to_s(options)}") end def to_svg(options = self.options) - linear_svg_gradient(color_stops, position_and_angle || Sass::Script::String.new("top")) + linear_svg_gradient(color_stops, position_or_angle || Sass::Script::String.new("top")) end def to_pie(options = self.options) # PIE just uses the standard rep, but the property is prefixed so @@ -150,13 +168,18 @@ module Compass::SassExtensions::Functions::GradientSupport module Functions # given a position list, return a corresponding position in percents + # otherwise, returns the original argument def grad_point(position) + original_value = position position = unless position.is_a?(Sass::Script::List) Sass::Script::List.new([position], :space) else Sass::Script::List.new(position.value.dup, position.separator) end - position.value.reject!{|p| p.is_a?(Sass::Script::Number) && p.numerator_units.include?("deg")} + # Handle unknown arguments by passing them along untouched. + unless position.value.all?{|p| is_position(p).to_bool } + return original_value + end if (position.value.first.value =~ /top|bottom/) or (position.value.last.value =~ /left|right/) # browsers are pretty forgiving of reversed positions so we are too. position.value.reverse! @@ -198,7 +221,7 @@ module Compass::SassExtensions::Functions::GradientSupport end, :comma) end - def radial_gradient(position_and_angle, shape_and_size, *color_stops) + def radial_gradient(position_or_angle, shape_and_size, *color_stops) # Have to deal with variable length/meaning arguments. if color_stop?(shape_and_size) color_stops.unshift(shape_and_size) @@ -209,38 +232,38 @@ module Compass::SassExtensions::Functions::GradientSupport shape_and_size = nil end shape_and_size = nil if shape_and_size && !shape_and_size.to_bool # nil out explictly passed falses - # ditto for position_and_angle - if color_stop?(position_and_angle) - color_stops.unshift(position_and_angle) - position_and_angle = nil - elsif list_of_color_stops?(position_and_angle) - color_stops = position_and_angle.value + color_stops - position_and_angle = nil + # ditto for position_or_angle + if color_stop?(position_or_angle) + color_stops.unshift(position_or_angle) + position_or_angle = nil + elsif list_of_color_stops?(position_or_angle) + color_stops = position_or_angle.value + color_stops + position_or_angle = nil end - position_and_angle = nil if position_and_angle && !position_and_angle.to_bool + position_or_angle = nil if position_or_angle && !position_or_angle.to_bool # Support legacy use of the color-stops() function if color_stops.size == 1 && list_of_color_stops?(color_stops.first) color_stops = color_stops.first.value end - RadialGradient.new(position_and_angle, shape_and_size, send(:color_stops, *color_stops)) + RadialGradient.new(position_or_angle, shape_and_size, send(:color_stops, *color_stops)) end - def linear_gradient(position_and_angle, *color_stops) - if color_stop?(position_and_angle) - color_stops.unshift(position_and_angle) - position_and_angle = nil - elsif list_of_color_stops?(position_and_angle) - color_stops = position_and_angle.value + color_stops - position_and_angle = nil + def linear_gradient(position_or_angle, *color_stops) + if color_stop?(position_or_angle) + color_stops.unshift(position_or_angle) + position_or_angle = nil + elsif list_of_color_stops?(position_or_angle) + color_stops = position_or_angle.value + color_stops + position_or_angle = nil end - position_and_angle = nil if position_and_angle && !position_and_angle.to_bool + position_or_angle = nil if position_or_angle && !position_or_angle.to_bool # Support legacy use of the color-stops() function if color_stops.size == 1 && (stops = list_of_color_stops?(color_stops.first)) color_stops = stops end - LinearGradient.new(position_and_angle, send(:color_stops, *color_stops)) + LinearGradient.new(position_or_angle, send(:color_stops, *color_stops)) end # returns color-stop() calls for use in webkit. diff --git a/test/fixtures/stylesheets/compass/css/gradients.css b/test/fixtures/stylesheets/compass/css/gradients.css index b8c6c077..2fd5b247 100644 --- a/test/fixtures/stylesheets/compass/css/gradients.css +++ b/test/fixtures/stylesheets/compass/css/gradients.css @@ -43,20 +43,16 @@ background-image: radial-gradient(center center, #dddddd, #aaaaaa 100px); } .bg-linear-gradient-with-angle { - background-image: url(''); - background-size: 100%; - background-image: -webkit-gradient(linear, 0% 0%, 100% 100%, color-stop(0%, #dddddd), color-stop(100%, #aaaaaa)); - background-image: -moz-linear-gradient(top left -45deg, #dddddd, #aaaaaa); - background-image: -o-linear-gradient(top left -45deg, #dddddd, #aaaaaa); - background-image: linear-gradient(top left -45deg, #dddddd, #aaaaaa); } + background-image: -webkit-gradient(linear, -45deg, -45deg, color-stop(0%, #dddddd), color-stop(100%, #aaaaaa)); + background-image: -moz-linear-gradient(-45deg, #dddddd, #aaaaaa); + background-image: -o-linear-gradient(-45deg, #dddddd, #aaaaaa); + background-image: linear-gradient(-45deg, #dddddd, #aaaaaa); } .bg-radial-gradient-with-angle-and-shape { - background-image: url(''); - background-size: 100%; - background-image: -webkit-gradient(radial, 50% 50%, 0, 50% 50%, 100, color-stop(0%, #dddddd), color-stop(100%, #aaaaaa)); - background-image: -moz-radial-gradient(center center 45deg, ellipse cover, #dddddd, #aaaaaa 100px); - background-image: -o-radial-gradient(center center 45deg, ellipse cover, #dddddd, #aaaaaa 100px); - background-image: radial-gradient(center center 45deg, ellipse cover, #dddddd, #aaaaaa 100px); } + background-image: -webkit-gradient(radial, 45deg, 0, 45deg, 100, color-stop(0%, #dddddd), color-stop(100%, #aaaaaa)); + background-image: -moz-radial-gradient(45deg, ellipse cover, #dddddd, #aaaaaa 100px); + background-image: -o-radial-gradient(45deg, ellipse cover, #dddddd, #aaaaaa 100px); + background-image: radial-gradient(45deg, ellipse cover, #dddddd, #aaaaaa 100px); } .bg-all-gradient-types { background-image: url('/images/4x6.png?busted=true'), url(''), url(''); diff --git a/test/fixtures/stylesheets/compass/sass/gradients.sass b/test/fixtures/stylesheets/compass/sass/gradients.sass index f521aede..747f25b3 100644 --- a/test/fixtures/stylesheets/compass/sass/gradients.sass +++ b/test/fixtures/stylesheets/compass/sass/gradients.sass @@ -23,10 +23,10 @@ $experimental-support-for-svg: true +background-image(radial-gradient(center center, #ddd, #aaa 100px)) .bg-linear-gradient-with-angle - +background-image(linear-gradient(top left -45deg, #ddd, #aaa)) + +background-image(linear-gradient(-45deg, #ddd, #aaa)) .bg-radial-gradient-with-angle-and-shape - +background-image(radial-gradient(center center 45deg, ellipse cover, #ddd, #aaa 100px)) + +background-image(radial-gradient(45deg, ellipse cover, #ddd, #aaa 100px)) .bg-all-gradient-types +background-image(image-url("4x6.png"), linear-gradient(top left, #ddd, #aaa), radial-gradient(center center, #ddd, #aaa 100px))