sass 3.4.0 → 3.4.25
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.yardopts +3 -1
- data/CODE_OF_CONDUCT.md +10 -0
- data/CONTRIBUTING.md +148 -0
- data/MIT-LICENSE +1 -1
- data/README.md +26 -20
- data/Rakefile +103 -20
- data/VERSION +1 -1
- data/VERSION_DATE +1 -1
- data/extra/sass-spec-ref.sh +32 -0
- data/extra/update_watch.rb +1 -1
- data/lib/sass/cache_stores/filesystem.rb +7 -7
- data/lib/sass/cache_stores/memory.rb +4 -5
- data/lib/sass/callbacks.rb +2 -2
- data/lib/sass/css.rb +11 -10
- data/lib/sass/deprecation.rb +55 -0
- data/lib/sass/engine.rb +83 -38
- data/lib/sass/environment.rb +26 -2
- data/lib/sass/error.rb +12 -12
- data/lib/sass/exec/base.rb +15 -3
- data/lib/sass/exec/sass_convert.rb +34 -15
- data/lib/sass/exec/sass_scss.rb +23 -7
- data/lib/sass/features.rb +2 -2
- data/lib/sass/importers/base.rb +1 -1
- data/lib/sass/importers/deprecated_path.rb +51 -0
- data/lib/sass/importers/filesystem.rb +24 -16
- data/lib/sass/importers.rb +1 -0
- data/lib/sass/logger/base.rb +8 -2
- data/lib/sass/logger/delayed.rb +50 -0
- data/lib/sass/logger.rb +8 -3
- data/lib/sass/plugin/compiler.rb +42 -25
- data/lib/sass/plugin/configuration.rb +38 -22
- data/lib/sass/plugin/merb.rb +2 -2
- data/lib/sass/plugin/rack.rb +3 -3
- data/lib/sass/plugin/rails.rb +1 -1
- data/lib/sass/plugin/staleness_checker.rb +3 -3
- data/lib/sass/plugin.rb +3 -2
- data/lib/sass/script/css_parser.rb +2 -3
- data/lib/sass/script/css_variable_warning.rb +52 -0
- data/lib/sass/script/functions.rb +140 -73
- data/lib/sass/script/lexer.rb +37 -22
- data/lib/sass/script/parser.rb +235 -40
- data/lib/sass/script/tree/funcall.rb +12 -5
- data/lib/sass/script/tree/interpolation.rb +109 -4
- data/lib/sass/script/tree/list_literal.rb +31 -4
- data/lib/sass/script/tree/literal.rb +4 -0
- data/lib/sass/script/tree/node.rb +21 -3
- data/lib/sass/script/tree/operation.rb +54 -1
- data/lib/sass/script/tree/string_interpolation.rb +58 -37
- data/lib/sass/script/tree/variable.rb +1 -1
- data/lib/sass/script/value/base.rb +10 -9
- data/lib/sass/script/value/color.rb +42 -24
- data/lib/sass/script/value/helpers.rb +16 -6
- data/lib/sass/script/value/map.rb +1 -1
- data/lib/sass/script/value/number.rb +52 -19
- data/lib/sass/script/value/string.rb +46 -5
- data/lib/sass/script.rb +3 -3
- data/lib/sass/scss/css_parser.rb +16 -2
- data/lib/sass/scss/parser.rb +120 -75
- data/lib/sass/scss/rx.rb +9 -10
- data/lib/sass/scss/static_parser.rb +19 -14
- data/lib/sass/scss.rb +0 -2
- data/lib/sass/selector/abstract_sequence.rb +8 -6
- data/lib/sass/selector/comma_sequence.rb +25 -9
- data/lib/sass/selector/pseudo.rb +45 -35
- data/lib/sass/selector/sequence.rb +54 -18
- data/lib/sass/selector/simple.rb +11 -11
- data/lib/sass/selector/simple_sequence.rb +34 -15
- data/lib/sass/selector.rb +7 -10
- data/lib/sass/shared.rb +1 -1
- data/lib/sass/source/map.rb +7 -4
- data/lib/sass/source/position.rb +4 -4
- data/lib/sass/stack.rb +2 -2
- data/lib/sass/supports.rb +8 -10
- data/lib/sass/tree/comment_node.rb +1 -1
- data/lib/sass/tree/css_import_node.rb +9 -1
- data/lib/sass/tree/function_node.rb +8 -3
- data/lib/sass/tree/import_node.rb +6 -5
- data/lib/sass/tree/node.rb +5 -3
- data/lib/sass/tree/prop_node.rb +5 -6
- data/lib/sass/tree/rule_node.rb +14 -4
- data/lib/sass/tree/visitors/check_nesting.rb +18 -22
- data/lib/sass/tree/visitors/convert.rb +43 -26
- data/lib/sass/tree/visitors/cssize.rb +5 -1
- data/lib/sass/tree/visitors/deep_copy.rb +1 -1
- data/lib/sass/tree/visitors/extend.rb +15 -13
- data/lib/sass/tree/visitors/perform.rb +42 -17
- data/lib/sass/tree/visitors/set_options.rb +1 -1
- data/lib/sass/tree/visitors/to_css.rb +58 -30
- data/lib/sass/util/multibyte_string_scanner.rb +0 -2
- data/lib/sass/util/normalized_map.rb +0 -1
- data/lib/sass/util/subset_map.rb +1 -2
- data/lib/sass/util.rb +125 -68
- data/lib/sass/version.rb +2 -2
- data/lib/sass.rb +10 -3
- data/test/sass/compiler_test.rb +6 -2
- data/test/sass/conversion_test.rb +187 -53
- data/test/sass/css2sass_test.rb +50 -1
- data/test/sass/css_variable_test.rb +132 -0
- data/test/sass/engine_test.rb +207 -61
- data/test/sass/exec_test.rb +10 -0
- data/test/sass/extend_test.rb +101 -29
- data/test/sass/functions_test.rb +60 -9
- data/test/sass/importer_test.rb +9 -0
- data/test/sass/more_templates/more1.sass +10 -10
- data/test/sass/more_templates/more_import.sass +2 -2
- data/test/sass/plugin_test.rb +10 -8
- data/test/sass/results/script.css +3 -3
- data/test/sass/script_conversion_test.rb +58 -29
- data/test/sass/script_test.rb +430 -53
- data/test/sass/scss/css_test.rb +73 -7
- data/test/sass/scss/rx_test.rb +4 -0
- data/test/sass/scss/scss_test.rb +309 -4
- data/test/sass/source_map_test.rb +152 -74
- data/test/sass/superselector_test.rb +19 -0
- data/test/sass/templates/_partial.sass +1 -1
- data/test/sass/templates/basic.sass +10 -10
- data/test/sass/templates/bork1.sass +1 -1
- data/test/sass/templates/bork5.sass +1 -1
- data/test/sass/templates/compact.sass +10 -10
- data/test/sass/templates/complex.sass +187 -187
- data/test/sass/templates/compressed.sass +10 -10
- data/test/sass/templates/expanded.sass +10 -10
- data/test/sass/templates/import.sass +2 -2
- data/test/sass/templates/importee.sass +3 -3
- data/test/sass/templates/mixins.sass +22 -22
- data/test/sass/templates/multiline.sass +4 -4
- data/test/sass/templates/nested.sass +13 -13
- data/test/sass/templates/parent_ref.sass +12 -12
- data/test/sass/templates/script.sass +70 -70
- data/test/sass/templates/subdir/nested_subdir/_nested_partial.sass +1 -1
- data/test/sass/templates/subdir/nested_subdir/nested_subdir.sass +2 -2
- data/test/sass/templates/subdir/subdir.sass +3 -3
- data/test/sass/templates/units.sass +10 -10
- data/test/sass/util/multibyte_string_scanner_test.rb +10 -2
- data/test/sass/util_test.rb +15 -44
- data/test/sass-spec.yml +3 -0
- data/test/test_helper.rb +5 -4
- metadata +302 -295
- data/CONTRIBUTING +0 -3
- data/lib/sass/scss/script_lexer.rb +0 -15
- data/lib/sass/scss/script_parser.rb +0 -25
@@ -1,5 +1,5 @@
|
|
1
1
|
require 'sass/script/functions'
|
2
|
-
require 'sass/util
|
2
|
+
require 'sass/util'
|
3
3
|
|
4
4
|
module Sass::Script::Tree
|
5
5
|
# A SassScript parse node representing a function call.
|
@@ -128,12 +128,15 @@ module Sass::Script::Tree
|
|
128
128
|
splat = Sass::Tree::Visitors::Perform.perform_splat(
|
129
129
|
@splat, keywords, @kwarg_splat, environment)
|
130
130
|
if (fn = environment.function(@name))
|
131
|
+
css_variable_warning.warn! if css_variable_warning
|
131
132
|
return without_original(perform_sass_fn(fn, args, splat, environment))
|
132
133
|
end
|
133
134
|
|
134
135
|
args = construct_ruby_args(ruby_name, args, splat, environment)
|
135
136
|
|
136
137
|
if Sass::Script::Functions.callable?(ruby_name)
|
138
|
+
css_variable_warning.warn! if css_variable_warning
|
139
|
+
|
137
140
|
local_environment = Sass::Environment.new(environment.global_env, environment.options)
|
138
141
|
local_environment.caller = Sass::ReadOnlyEnvironment.new(environment, environment.options)
|
139
142
|
result = opts(Sass::Script::Functions::EvaluationContext.new(
|
@@ -208,7 +211,7 @@ module Sass::Script::Tree
|
|
208
211
|
|
209
212
|
argnames = signature.args[args.size..-1] || []
|
210
213
|
deprecated_argnames = (signature.deprecated && signature.deprecated[args.size..-1]) || []
|
211
|
-
args
|
214
|
+
args += argnames.zip(deprecated_argnames).map do |(argname, deprecated_argname)|
|
212
215
|
if keywords.has_key?(argname)
|
213
216
|
keywords.delete(argname)
|
214
217
|
elsif deprecated_argname && keywords.has_key?(deprecated_argname)
|
@@ -241,7 +244,7 @@ module Sass::Script::Tree
|
|
241
244
|
end
|
242
245
|
|
243
246
|
def perform_sass_fn(function, args, splat, environment)
|
244
|
-
Sass::Tree::Visitors::Perform.perform_arguments(function, args, splat) do |env|
|
247
|
+
Sass::Tree::Visitors::Perform.perform_arguments(function, args, splat, environment) do |env|
|
245
248
|
env.caller = Sass::Environment.new(environment)
|
246
249
|
|
247
250
|
val = catch :_sass_return do
|
@@ -296,8 +299,12 @@ module Sass::Script::Tree
|
|
296
299
|
message = "wrong number of arguments (#{given} for #{expected})"
|
297
300
|
end
|
298
301
|
end
|
299
|
-
elsif
|
300
|
-
|
302
|
+
elsif (md = /^wrong number of arguments \(given (\d+), expected (\d+)\)/.match(e.message)) &&
|
303
|
+
e.backtrace[0] =~ /:in `#{ruby_name}'$/
|
304
|
+
# Handle ruby 2.3 error formatting
|
305
|
+
message = "wrong number of arguments (#{md[1]} for #{md[2]})"
|
306
|
+
elsif e.message =~ /^wrong number of arguments/ &&
|
307
|
+
e.backtrace[0] !~ /:in `(block in )?#{ruby_name}'$/
|
301
308
|
raise e
|
302
309
|
end
|
303
310
|
raise Sass::SyntaxError.new("#{message} for `#{name}'")
|
@@ -27,6 +27,17 @@ module Sass::Script::Tree
|
|
27
27
|
# generate a warning.
|
28
28
|
attr_reader :warn_for_color
|
29
29
|
|
30
|
+
# The type of interpolation deprecation for this node.
|
31
|
+
#
|
32
|
+
# This can be `:none`, indicating that the node doesn't use deprecated
|
33
|
+
# interpolation; `:immediate`, indicating that a deprecation warning should
|
34
|
+
# be emitted as soon as possible; or `:potential`, indicating that a
|
35
|
+
# deprecation warning should be emitted if the resulting string is used in a
|
36
|
+
# way that would distinguish it from a list.
|
37
|
+
#
|
38
|
+
# @return [Symbol]
|
39
|
+
attr_reader :deprecation
|
40
|
+
|
30
41
|
# Interpolation in a property is of the form `before #{mid} after`.
|
31
42
|
#
|
32
43
|
# @param before [Node] See {Interpolation#before}
|
@@ -38,15 +49,16 @@ module Sass::Script::Tree
|
|
38
49
|
# @param warn_for_color [Boolean] See {Interpolation#warn_for_color}
|
39
50
|
# @comment
|
40
51
|
# rubocop:disable ParameterLists
|
41
|
-
def initialize(before, mid, after, wb, wa,
|
52
|
+
def initialize(before, mid, after, wb, wa, opts = {})
|
42
53
|
# rubocop:enable ParameterLists
|
43
54
|
@before = before
|
44
55
|
@mid = mid
|
45
56
|
@after = after
|
46
57
|
@whitespace_before = wb
|
47
58
|
@whitespace_after = wa
|
48
|
-
@originally_text = originally_text
|
49
|
-
@warn_for_color = warn_for_color
|
59
|
+
@originally_text = opts[:originally_text] || false
|
60
|
+
@warn_for_color = opts[:warn_for_color] || false
|
61
|
+
@deprecation = opts[:deprecation] || :none
|
50
62
|
end
|
51
63
|
|
52
64
|
# @return [String] A human-readable s-expression representation of the interpolation
|
@@ -56,6 +68,8 @@ module Sass::Script::Tree
|
|
56
68
|
|
57
69
|
# @see Node#to_sass
|
58
70
|
def to_sass(opts = {})
|
71
|
+
return to_quoted_equivalent.to_sass if deprecation == :immediate
|
72
|
+
|
59
73
|
res = ""
|
60
74
|
res << @before.to_sass(opts) if @before
|
61
75
|
res << ' ' if @before && @whitespace_before
|
@@ -67,6 +81,19 @@ module Sass::Script::Tree
|
|
67
81
|
res
|
68
82
|
end
|
69
83
|
|
84
|
+
# Returns an `unquote()` expression that will evaluate to the same value as
|
85
|
+
# this interpolation.
|
86
|
+
#
|
87
|
+
# @return [Sass::Script::Tree::Node]
|
88
|
+
def to_quoted_equivalent
|
89
|
+
Funcall.new(
|
90
|
+
"unquote",
|
91
|
+
[to_string_interpolation(self)],
|
92
|
+
Sass::Util::NormalizedMap.new,
|
93
|
+
nil,
|
94
|
+
nil)
|
95
|
+
end
|
96
|
+
|
70
97
|
# Returns the three components of the interpolation, `before`, `mid`, and `after`.
|
71
98
|
#
|
72
99
|
# @return [Array<Node>]
|
@@ -87,6 +114,49 @@ module Sass::Script::Tree
|
|
87
114
|
|
88
115
|
protected
|
89
116
|
|
117
|
+
# Converts a script node into a corresponding string interpolation
|
118
|
+
# expression.
|
119
|
+
#
|
120
|
+
# @param node_or_interp [Sass::Script::Tree::Node]
|
121
|
+
# @return [Sass::Script::Tree::StringInterpolation]
|
122
|
+
def to_string_interpolation(node_or_interp)
|
123
|
+
unless node_or_interp.is_a?(Interpolation)
|
124
|
+
node = node_or_interp
|
125
|
+
return string_literal(node.value.to_s) if node.is_a?(Literal)
|
126
|
+
if node.is_a?(StringInterpolation)
|
127
|
+
return concat(string_literal(node.quote), concat(node, string_literal(node.quote)))
|
128
|
+
end
|
129
|
+
return StringInterpolation.new(string_literal(""), node, string_literal(""))
|
130
|
+
end
|
131
|
+
|
132
|
+
interp = node_or_interp
|
133
|
+
after_string_or_interp =
|
134
|
+
if interp.after
|
135
|
+
to_string_interpolation(interp.after)
|
136
|
+
else
|
137
|
+
string_literal("")
|
138
|
+
end
|
139
|
+
if interp.after && interp.whitespace_after
|
140
|
+
after_string_or_interp = concat(string_literal(' '), after_string_or_interp)
|
141
|
+
end
|
142
|
+
|
143
|
+
mid_string_or_interp = to_string_interpolation(interp.mid)
|
144
|
+
|
145
|
+
before_string_or_interp =
|
146
|
+
if interp.before
|
147
|
+
to_string_interpolation(interp.before)
|
148
|
+
else
|
149
|
+
string_literal("")
|
150
|
+
end
|
151
|
+
if interp.before && interp.whitespace_before
|
152
|
+
before_string_or_interp = concat(before_string_or_interp, string_literal(' '))
|
153
|
+
end
|
154
|
+
|
155
|
+
concat(before_string_or_interp, concat(mid_string_or_interp, after_string_or_interp))
|
156
|
+
end
|
157
|
+
|
158
|
+
private
|
159
|
+
|
90
160
|
# Evaluates the interpolation.
|
91
161
|
#
|
92
162
|
# @param environment [Sass::Environment] The environment in which to evaluate the SassScript
|
@@ -112,7 +182,42 @@ MESSAGE
|
|
112
182
|
res << val.to_s(:quote => :none)
|
113
183
|
res << " " if @after && @whitespace_after
|
114
184
|
res << @after.perform(environment).to_s if @after
|
115
|
-
|
185
|
+
str = Sass::Script::Value::String.new(
|
186
|
+
res, :identifier,
|
187
|
+
(to_quoted_equivalent.to_sass if deprecation == :potential))
|
188
|
+
str.source_range = source_range
|
189
|
+
opts(str)
|
190
|
+
end
|
191
|
+
|
192
|
+
# Concatenates two string literals or string interpolation expressions.
|
193
|
+
#
|
194
|
+
# @param string_or_interp1 [Sass::Script::Tree::Literal|Sass::Script::Tree::StringInterpolation]
|
195
|
+
# @param string_or_interp2 [Sass::Script::Tree::Literal|Sass::Script::Tree::StringInterpolation]
|
196
|
+
# @return [Sass::Script::Tree::StringInterpolation]
|
197
|
+
def concat(string_or_interp1, string_or_interp2)
|
198
|
+
if string_or_interp1.is_a?(Literal) && string_or_interp2.is_a?(Literal)
|
199
|
+
return string_literal(string_or_interp1.value.value + string_or_interp2.value.value)
|
200
|
+
end
|
201
|
+
|
202
|
+
if string_or_interp1.is_a?(Literal)
|
203
|
+
string = string_or_interp1
|
204
|
+
interp = string_or_interp2
|
205
|
+
before = string_literal(string.value.value + interp.before.value.value)
|
206
|
+
return StringInterpolation.new(before, interp.mid, interp.after)
|
207
|
+
end
|
208
|
+
|
209
|
+
StringInterpolation.new(
|
210
|
+
string_or_interp1.before,
|
211
|
+
string_or_interp1.mid,
|
212
|
+
concat(string_or_interp1.after, string_or_interp2))
|
213
|
+
end
|
214
|
+
|
215
|
+
# Returns a string literal with the given contents.
|
216
|
+
#
|
217
|
+
# @param string [String]
|
218
|
+
# @return string [Sass::Script::Tree::Literal]
|
219
|
+
def string_literal(string)
|
220
|
+
Literal.new(Sass::Script::Value::String.new(string, :string))
|
116
221
|
end
|
117
222
|
end
|
118
223
|
end
|
@@ -28,11 +28,8 @@ module Sass::Script::Tree
|
|
28
28
|
# @see Value#to_sass
|
29
29
|
def to_sass(opts = {})
|
30
30
|
return "()" if elements.empty?
|
31
|
-
precedence = Sass::Script::Parser.precedence_of(separator)
|
32
31
|
members = elements.map do |v|
|
33
|
-
if
|
34
|
-
separator == :space && v.is_a?(UnaryOperation) &&
|
35
|
-
(v.operator == :minus || v.operator == :plus)
|
32
|
+
if element_needs_parens?(v)
|
36
33
|
"(#{v.to_sass(opts)})"
|
37
34
|
else
|
38
35
|
v.to_sass(opts)
|
@@ -55,6 +52,10 @@ module Sass::Script::Tree
|
|
55
52
|
"(#{elements.map {|e| e.inspect}.join(separator == :space ? ' ' : ', ')})"
|
56
53
|
end
|
57
54
|
|
55
|
+
def force_division!
|
56
|
+
# Do nothing. Lists prevent division propagation.
|
57
|
+
end
|
58
|
+
|
58
59
|
protected
|
59
60
|
|
60
61
|
def _perform(environment)
|
@@ -68,6 +69,32 @@ module Sass::Script::Tree
|
|
68
69
|
|
69
70
|
private
|
70
71
|
|
72
|
+
# Returns whether an element in the list should be wrapped in parentheses
|
73
|
+
# when serialized to Sass.
|
74
|
+
def element_needs_parens?(element)
|
75
|
+
if element.is_a?(ListLiteral)
|
76
|
+
return Sass::Script::Parser.precedence_of(element.separator) <=
|
77
|
+
Sass::Script::Parser.precedence_of(separator)
|
78
|
+
end
|
79
|
+
|
80
|
+
return false unless separator == :space
|
81
|
+
|
82
|
+
if element.is_a?(UnaryOperation)
|
83
|
+
return element.operator == :minus || element.operator == :plus
|
84
|
+
end
|
85
|
+
|
86
|
+
return false unless element.is_a?(Operation)
|
87
|
+
return true unless element.operator == :div
|
88
|
+
!(is_literal_number?(element.operand1) && is_literal_number?(element.operand2))
|
89
|
+
end
|
90
|
+
|
91
|
+
# Returns whether a value is a number literal that shouldn't be divided.
|
92
|
+
def is_literal_number?(value)
|
93
|
+
value.is_a?(Literal) &&
|
94
|
+
value.value.is_a?((Sass::Script::Value::Number)) &&
|
95
|
+
!value.value.original.nil?
|
96
|
+
end
|
97
|
+
|
71
98
|
def sep_str(opts = options)
|
72
99
|
return ' ' if separator == :space
|
73
100
|
return ',' if opts && opts[:style] == :compressed
|
@@ -10,7 +10,7 @@ module Sass::Script::Tree
|
|
10
10
|
|
11
11
|
# The line of the document on which this node appeared.
|
12
12
|
#
|
13
|
-
# @return [
|
13
|
+
# @return [Integer]
|
14
14
|
attr_accessor :line
|
15
15
|
|
16
16
|
# The source range in the document on which this node appeared.
|
@@ -23,9 +23,17 @@ module Sass::Script::Tree
|
|
23
23
|
# @return [String]
|
24
24
|
attr_accessor :filename
|
25
25
|
|
26
|
+
# The warning that this node should emit if it executes in a way that's not
|
27
|
+
# safe for a CSS variable value.
|
28
|
+
#
|
29
|
+
# This is `nil` if this is not in a CSS variable value.
|
30
|
+
#
|
31
|
+
# @return [Sass::Script::CssVariableWarning]
|
32
|
+
attr_accessor :css_variable_warning
|
33
|
+
|
26
34
|
# Sets the options hash for this node,
|
27
35
|
# as well as for all child nodes.
|
28
|
-
# See {file:SASS_REFERENCE.md#
|
36
|
+
# See {file:SASS_REFERENCE.md#Options the Sass options documentation}.
|
29
37
|
#
|
30
38
|
# @param options [{Symbol => Object}] The options
|
31
39
|
def options=(options)
|
@@ -62,6 +70,10 @@ module Sass::Script::Tree
|
|
62
70
|
|
63
71
|
# Returns the text of this SassScript expression.
|
64
72
|
#
|
73
|
+
# @options opts :quote [String]
|
74
|
+
# The preferred quote style for quoted strings. If `:none`, strings are
|
75
|
+
# always emitted unquoted.
|
76
|
+
#
|
65
77
|
# @return [String]
|
66
78
|
def to_sass(opts = {})
|
67
79
|
Sass::Util.abstract(self)
|
@@ -75,12 +87,18 @@ module Sass::Script::Tree
|
|
75
87
|
Sass::Util.abstract(self)
|
76
88
|
end
|
77
89
|
|
90
|
+
# Forces any division operations with number literals in this expression to
|
91
|
+
# do real division, rather than returning strings.
|
92
|
+
def force_division!
|
93
|
+
children.each {|c| c.force_division!}
|
94
|
+
end
|
95
|
+
|
78
96
|
protected
|
79
97
|
|
80
98
|
# Converts underscores to dashes if the :dasherize option is set.
|
81
99
|
def dasherize(s, opts)
|
82
100
|
if opts[:dasherize]
|
83
|
-
s.
|
101
|
+
s.tr('_', '-')
|
84
102
|
else
|
85
103
|
s
|
86
104
|
end
|
@@ -2,6 +2,9 @@ module Sass::Script::Tree
|
|
2
2
|
# A SassScript parse node representing a binary operation,
|
3
3
|
# such as `$a + $b` or `"foo" + 1`.
|
4
4
|
class Operation < Node
|
5
|
+
@@color_arithmetic_deprecation = Sass::Deprecation.new
|
6
|
+
@@unitless_equals_deprecation = Sass::Deprecation.new
|
7
|
+
|
5
8
|
attr_reader :operand1
|
6
9
|
attr_reader :operand2
|
7
10
|
attr_reader :operator
|
@@ -78,16 +81,66 @@ module Sass::Script::Tree
|
|
78
81
|
"Invalid null operation: \"#{value1.inspect} #{@operator} #{value2.inspect}\".")
|
79
82
|
end
|
80
83
|
|
84
|
+
if css_variable_warning && @operator == :div &&
|
85
|
+
!(value1.is_a?(Sass::Script::Value::Number) && value1.original &&
|
86
|
+
value2.is_a?(Sass::Script::Value::Number) && value2.original) &&
|
87
|
+
!(value1.is_a?(Sass::Script::Value::String) && value2.is_a?(Sass::Script::Value::String))
|
88
|
+
css_variable_warning.warn!
|
89
|
+
end
|
90
|
+
|
81
91
|
begin
|
82
|
-
opts(value1.send(@operator, value2))
|
92
|
+
result = opts(value1.send(@operator, value2))
|
83
93
|
rescue NoMethodError => e
|
84
94
|
raise e unless e.name.to_s == @operator.to_s
|
85
95
|
raise Sass::SyntaxError.new("Undefined operation: \"#{value1} #{@operator} #{value2}\".")
|
86
96
|
end
|
97
|
+
|
98
|
+
warn_for_color_arithmetic(value1, value2)
|
99
|
+
warn_for_unitless_equals(value1, value2, result)
|
100
|
+
|
101
|
+
result
|
87
102
|
end
|
88
103
|
|
89
104
|
private
|
90
105
|
|
106
|
+
def warn_for_color_arithmetic(value1, value2)
|
107
|
+
return unless @operator == :plus || @operator == :times || @operator == :minus ||
|
108
|
+
@operator == :div || @operator == :mod
|
109
|
+
|
110
|
+
if value1.is_a?(Sass::Script::Value::Number)
|
111
|
+
return unless value2.is_a?(Sass::Script::Value::Color)
|
112
|
+
elsif value1.is_a?(Sass::Script::Value::Color)
|
113
|
+
return unless value2.is_a?(Sass::Script::Value::Color) || value2.is_a?(Sass::Script::Value::Number)
|
114
|
+
else
|
115
|
+
return
|
116
|
+
end
|
117
|
+
|
118
|
+
@@color_arithmetic_deprecation.warn(filename, line, <<WARNING)
|
119
|
+
The operation `#{value1} #{@operator} #{value2}` is deprecated and will be an error in future versions.
|
120
|
+
Consider using Sass's color functions instead.
|
121
|
+
http://sass-lang.com/documentation/Sass/Script/Functions.html#other_color_functions
|
122
|
+
WARNING
|
123
|
+
end
|
124
|
+
|
125
|
+
def warn_for_unitless_equals(value1, value2, result)
|
126
|
+
return unless @operator == :eq || @operator == :neq
|
127
|
+
return unless value1.is_a?(Sass::Script::Value::Number)
|
128
|
+
return unless value2.is_a?(Sass::Script::Value::Number)
|
129
|
+
return unless value1.unitless? != value2.unitless?
|
130
|
+
return unless result == (if @operator == :eq
|
131
|
+
Sass::Script::Value::Bool::TRUE
|
132
|
+
else
|
133
|
+
Sass::Script::Value::Bool::FALSE
|
134
|
+
end)
|
135
|
+
|
136
|
+
operation = "#{value1.to_sass} #{@operator == :eq ? '==' : '!='} #{value2.to_sass}"
|
137
|
+
future_value = @operator == :neq
|
138
|
+
@@unitless_equals_deprecation.warn(filename, line, <<WARNING)
|
139
|
+
The result of `#{operation}` will be `#{future_value}` in future releases of Sass.
|
140
|
+
Unitless numbers will no longer be equal to the same numbers with units.
|
141
|
+
WARNING
|
142
|
+
end
|
143
|
+
|
91
144
|
def operand_to_sass(op, side, opts)
|
92
145
|
return "(#{op.to_sass(opts)})" if op.is_a?(Sass::Script::Tree::ListLiteral)
|
93
146
|
return op.to_sass(opts) unless op.is_a?(Operation)
|
@@ -3,12 +3,39 @@ module Sass::Script::Tree
|
|
3
3
|
#
|
4
4
|
# @see Interpolation
|
5
5
|
class StringInterpolation < Node
|
6
|
+
# @return [Literal] The string literal before this interpolation.
|
7
|
+
attr_reader :before
|
8
|
+
|
9
|
+
# @return [Node] The SassScript within the interpolation
|
10
|
+
attr_reader :mid
|
11
|
+
|
12
|
+
# @return [StringInterpolation, Literal]
|
13
|
+
# The string literal or string interpolation before this interpolation.
|
14
|
+
attr_reader :after
|
15
|
+
|
16
|
+
# Whether this is a CSS string or a CSS identifier. The difference is that
|
17
|
+
# strings are written with double-quotes, while identifiers aren't.
|
18
|
+
#
|
19
|
+
# String interpolations are only ever identifiers if they're quote-like
|
20
|
+
# functions such as `url()`.
|
21
|
+
#
|
22
|
+
# @return [Symbol] `:string` or `:identifier`
|
23
|
+
def type
|
24
|
+
@before.value.type
|
25
|
+
end
|
26
|
+
|
27
|
+
# Returns the quote character that should be used to wrap a Sass
|
28
|
+
# representation of this interpolation.
|
29
|
+
def quote
|
30
|
+
quote_for(self) || '"'
|
31
|
+
end
|
32
|
+
|
6
33
|
# Interpolation in a string is of the form `"before #{mid} after"`,
|
7
34
|
# where `before` and `after` may include more interpolation.
|
8
35
|
#
|
9
|
-
# @param before [
|
10
|
-
# @param mid [Node]
|
11
|
-
# @param after [
|
36
|
+
# @param before [StringInterpolation, Literal] See {StringInterpolation#before}
|
37
|
+
# @param mid [Node] See {StringInterpolation#mid}
|
38
|
+
# @param after [Literal] See {StringInterpolation#after}
|
12
39
|
def initialize(before, mid, after)
|
13
40
|
@before = before
|
14
41
|
@mid = mid
|
@@ -22,32 +49,15 @@ module Sass::Script::Tree
|
|
22
49
|
|
23
50
|
# @see Node#to_sass
|
24
51
|
def to_sass(opts = {})
|
25
|
-
|
26
|
-
|
27
|
-
before_unquote, before_quote_char, before_str = parse_str(@before.to_sass(opts))
|
28
|
-
after_unquote, after_quote_char, after_str = parse_str(@after.to_sass(opts))
|
29
|
-
unquote = before_unquote || after_unquote ||
|
30
|
-
(before_quote_char && !after_quote_char && !after_str.empty?) ||
|
31
|
-
(!before_quote_char && after_quote_char && !before_str.empty?)
|
32
|
-
quote_char =
|
33
|
-
if before_quote_char && after_quote_char && before_quote_char != after_quote_char
|
34
|
-
before_str.gsub!("\\'", "'")
|
35
|
-
before_str.gsub!('"', "\\\"")
|
36
|
-
after_str.gsub!("\\'", "'")
|
37
|
-
after_str.gsub!('"', "\\\"")
|
38
|
-
'"'
|
39
|
-
else
|
40
|
-
before_quote_char || after_quote_char
|
41
|
-
end
|
52
|
+
quote = type == :string ? opts[:quote] || quote_for(self) || '"' : :none
|
53
|
+
opts = opts.merge(:quote => quote)
|
42
54
|
|
43
55
|
res = ""
|
44
|
-
res <<
|
45
|
-
res <<
|
46
|
-
res <<
|
47
|
-
res <<
|
48
|
-
res <<
|
49
|
-
res << quote_char if quote_char
|
50
|
-
res << ')' if unquote
|
56
|
+
res << quote if quote != :none
|
57
|
+
res << _to_sass(before, opts)
|
58
|
+
res << '#{' << @mid.to_sass(opts.merge(:quote => nil)) << '}'
|
59
|
+
res << _to_sass(after, opts)
|
60
|
+
res << quote if quote != :none
|
51
61
|
res
|
52
62
|
end
|
53
63
|
|
@@ -88,17 +98,28 @@ module Sass::Script::Tree
|
|
88
98
|
|
89
99
|
private
|
90
100
|
|
91
|
-
def
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
return
|
99
|
-
|
100
|
-
return
|
101
|
+
def _to_sass(string_or_interp, opts)
|
102
|
+
result = string_or_interp.to_sass(opts)
|
103
|
+
opts[:quote] == :none ? result : result.slice(1...-1)
|
104
|
+
end
|
105
|
+
|
106
|
+
def quote_for(string_or_interp)
|
107
|
+
if string_or_interp.is_a?(Sass::Script::Tree::Literal)
|
108
|
+
return nil if string_or_interp.value.value.empty?
|
109
|
+
return '"' if string_or_interp.value.value.include?("'")
|
110
|
+
return "'" if string_or_interp.value.value.include?('"')
|
111
|
+
return nil
|
101
112
|
end
|
113
|
+
|
114
|
+
# Double-quotes take precedence over single quotes.
|
115
|
+
before_quote = quote_for(string_or_interp.before)
|
116
|
+
return '"' if before_quote == '"'
|
117
|
+
after_quote = quote_for(string_or_interp.after)
|
118
|
+
return '"' if after_quote == '"'
|
119
|
+
|
120
|
+
# Returns "'" if either or both insist on single quotes, and nil
|
121
|
+
# otherwise.
|
122
|
+
before_quote || after_quote
|
102
123
|
end
|
103
124
|
end
|
104
125
|
end
|
@@ -22,11 +22,12 @@ module Sass::Script::Value
|
|
22
22
|
def initialize(value = nil)
|
23
23
|
value.freeze unless value.nil? || value == true || value == false
|
24
24
|
@value = value
|
25
|
+
@options = nil
|
25
26
|
end
|
26
27
|
|
27
28
|
# Sets the options hash for this node,
|
28
29
|
# as well as for all child nodes.
|
29
|
-
# See {file:SASS_REFERENCE.md#
|
30
|
+
# See {file:SASS_REFERENCE.md#Options the Sass options documentation}.
|
30
31
|
#
|
31
32
|
# @param options [{Symbol => Object}] The options
|
32
33
|
attr_writer :options
|
@@ -87,7 +88,7 @@ MSG
|
|
87
88
|
# @return [Script::Value::String] A string containing both values
|
88
89
|
# separated by `"="`
|
89
90
|
def single_eq(other)
|
90
|
-
Sass::Script::Value::String.new("#{
|
91
|
+
Sass::Script::Value::String.new("#{self}=#{other}")
|
91
92
|
end
|
92
93
|
|
93
94
|
# The SassScript `+` operation.
|
@@ -106,7 +107,7 @@ MSG
|
|
106
107
|
# @return [Script::Value::String] A string containing both values
|
107
108
|
# separated by `"-"`
|
108
109
|
def minus(other)
|
109
|
-
Sass::Script::Value::String.new("#{
|
110
|
+
Sass::Script::Value::String.new("#{self}-#{other}")
|
110
111
|
end
|
111
112
|
|
112
113
|
# The SassScript `/` operation.
|
@@ -115,7 +116,7 @@ MSG
|
|
115
116
|
# @return [Script::Value::String] A string containing both values
|
116
117
|
# separated by `"/"`
|
117
118
|
def div(other)
|
118
|
-
Sass::Script::Value::String.new("#{
|
119
|
+
Sass::Script::Value::String.new("#{self}/#{other}")
|
119
120
|
end
|
120
121
|
|
121
122
|
# The SassScript unary `+` operation (e.g. `+$a`).
|
@@ -124,7 +125,7 @@ MSG
|
|
124
125
|
# @return [Script::Value::String] A string containing the value
|
125
126
|
# preceded by `"+"`
|
126
127
|
def unary_plus
|
127
|
-
Sass::Script::Value::String.new("+#{
|
128
|
+
Sass::Script::Value::String.new("+#{self}")
|
128
129
|
end
|
129
130
|
|
130
131
|
# The SassScript unary `-` operation (e.g. `-$a`).
|
@@ -133,7 +134,7 @@ MSG
|
|
133
134
|
# @return [Script::Value::String] A string containing the value
|
134
135
|
# preceded by `"-"`
|
135
136
|
def unary_minus
|
136
|
-
Sass::Script::Value::String.new("-#{
|
137
|
+
Sass::Script::Value::String.new("-#{self}")
|
137
138
|
end
|
138
139
|
|
139
140
|
# The SassScript unary `/` operation (e.g. `/$a`).
|
@@ -142,13 +143,13 @@ MSG
|
|
142
143
|
# @return [Script::Value::String] A string containing the value
|
143
144
|
# preceded by `"/"`
|
144
145
|
def unary_div
|
145
|
-
Sass::Script::Value::String.new("/#{
|
146
|
+
Sass::Script::Value::String.new("/#{self}")
|
146
147
|
end
|
147
148
|
|
148
149
|
# Returns the hash code of this value. Two objects' hash codes should be
|
149
150
|
# equal if the objects are equal.
|
150
151
|
#
|
151
|
-
# @return [Fixnum] The hash code.
|
152
|
+
# @return [Integer for Ruby 2.4.0+, Fixnum for earlier Ruby versions] The hash code.
|
152
153
|
def hash
|
153
154
|
value.hash
|
154
155
|
end
|
@@ -175,7 +176,7 @@ MSG
|
|
175
176
|
eq(other).to_bool
|
176
177
|
end
|
177
178
|
|
178
|
-
# @return [
|
179
|
+
# @return [Integer] The integer value of this value
|
179
180
|
# @raise [Sass::SyntaxError] if this value isn't an integer
|
180
181
|
def to_i
|
181
182
|
raise Sass::SyntaxError.new("#{inspect} is not an integer.")
|