xass 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.yardopts +11 -0
- data/CONTRIBUTING +3 -0
- data/MIT-LICENSE +20 -0
- data/README.md +201 -0
- data/Rakefile +349 -0
- data/VERSION +1 -0
- data/VERSION_NAME +1 -0
- data/bin/push +13 -0
- data/bin/sass +13 -0
- data/bin/sass-convert +12 -0
- data/bin/scss +13 -0
- data/extra/update_watch.rb +13 -0
- data/init.rb +18 -0
- data/lib/sass/cache_stores/base.rb +88 -0
- data/lib/sass/cache_stores/chain.rb +33 -0
- data/lib/sass/cache_stores/filesystem.rb +64 -0
- data/lib/sass/cache_stores/memory.rb +47 -0
- data/lib/sass/cache_stores/null.rb +25 -0
- data/lib/sass/cache_stores.rb +15 -0
- data/lib/sass/callbacks.rb +66 -0
- data/lib/sass/css.rb +409 -0
- data/lib/sass/engine.rb +930 -0
- data/lib/sass/environment.rb +101 -0
- data/lib/sass/error.rb +201 -0
- data/lib/sass/exec.rb +707 -0
- data/lib/sass/importers/base.rb +139 -0
- data/lib/sass/importers/filesystem.rb +186 -0
- data/lib/sass/importers.rb +22 -0
- data/lib/sass/logger/base.rb +32 -0
- data/lib/sass/logger/log_level.rb +49 -0
- data/lib/sass/logger.rb +15 -0
- data/lib/sass/media.rb +213 -0
- data/lib/sass/plugin/compiler.rb +406 -0
- data/lib/sass/plugin/configuration.rb +123 -0
- data/lib/sass/plugin/generic.rb +15 -0
- data/lib/sass/plugin/merb.rb +48 -0
- data/lib/sass/plugin/rack.rb +60 -0
- data/lib/sass/plugin/rails.rb +47 -0
- data/lib/sass/plugin/staleness_checker.rb +199 -0
- data/lib/sass/plugin.rb +133 -0
- data/lib/sass/railtie.rb +10 -0
- data/lib/sass/repl.rb +57 -0
- data/lib/sass/root.rb +7 -0
- data/lib/sass/script/arg_list.rb +52 -0
- data/lib/sass/script/bool.rb +18 -0
- data/lib/sass/script/color.rb +606 -0
- data/lib/sass/script/css_lexer.rb +29 -0
- data/lib/sass/script/css_parser.rb +31 -0
- data/lib/sass/script/funcall.rb +245 -0
- data/lib/sass/script/functions.rb +1543 -0
- data/lib/sass/script/interpolation.rb +79 -0
- data/lib/sass/script/lexer.rb +345 -0
- data/lib/sass/script/list.rb +85 -0
- data/lib/sass/script/literal.rb +221 -0
- data/lib/sass/script/node.rb +99 -0
- data/lib/sass/script/null.rb +37 -0
- data/lib/sass/script/number.rb +453 -0
- data/lib/sass/script/operation.rb +110 -0
- data/lib/sass/script/parser.rb +502 -0
- data/lib/sass/script/string.rb +51 -0
- data/lib/sass/script/string_interpolation.rb +103 -0
- data/lib/sass/script/unary_operation.rb +69 -0
- data/lib/sass/script/variable.rb +58 -0
- data/lib/sass/script.rb +39 -0
- data/lib/sass/scss/css_parser.rb +36 -0
- data/lib/sass/scss/parser.rb +1180 -0
- data/lib/sass/scss/rx.rb +133 -0
- data/lib/sass/scss/script_lexer.rb +15 -0
- data/lib/sass/scss/script_parser.rb +25 -0
- data/lib/sass/scss/static_parser.rb +54 -0
- data/lib/sass/scss.rb +16 -0
- data/lib/sass/selector/abstract_sequence.rb +94 -0
- data/lib/sass/selector/comma_sequence.rb +92 -0
- data/lib/sass/selector/sequence.rb +507 -0
- data/lib/sass/selector/simple.rb +119 -0
- data/lib/sass/selector/simple_sequence.rb +215 -0
- data/lib/sass/selector.rb +452 -0
- data/lib/sass/shared.rb +76 -0
- data/lib/sass/supports.rb +229 -0
- data/lib/sass/tree/charset_node.rb +22 -0
- data/lib/sass/tree/comment_node.rb +82 -0
- data/lib/sass/tree/content_node.rb +9 -0
- data/lib/sass/tree/css_import_node.rb +60 -0
- data/lib/sass/tree/debug_node.rb +18 -0
- data/lib/sass/tree/directive_node.rb +42 -0
- data/lib/sass/tree/each_node.rb +24 -0
- data/lib/sass/tree/extend_node.rb +36 -0
- data/lib/sass/tree/for_node.rb +36 -0
- data/lib/sass/tree/function_node.rb +34 -0
- data/lib/sass/tree/if_node.rb +52 -0
- data/lib/sass/tree/import_node.rb +75 -0
- data/lib/sass/tree/media_node.rb +58 -0
- data/lib/sass/tree/mixin_def_node.rb +38 -0
- data/lib/sass/tree/mixin_node.rb +39 -0
- data/lib/sass/tree/node.rb +196 -0
- data/lib/sass/tree/prop_node.rb +152 -0
- data/lib/sass/tree/return_node.rb +18 -0
- data/lib/sass/tree/root_node.rb +78 -0
- data/lib/sass/tree/rule_node.rb +132 -0
- data/lib/sass/tree/supports_node.rb +51 -0
- data/lib/sass/tree/trace_node.rb +32 -0
- data/lib/sass/tree/variable_node.rb +30 -0
- data/lib/sass/tree/visitors/base.rb +75 -0
- data/lib/sass/tree/visitors/check_nesting.rb +147 -0
- data/lib/sass/tree/visitors/convert.rb +316 -0
- data/lib/sass/tree/visitors/cssize.rb +241 -0
- data/lib/sass/tree/visitors/deep_copy.rb +102 -0
- data/lib/sass/tree/visitors/extend.rb +68 -0
- data/lib/sass/tree/visitors/perform.rb +446 -0
- data/lib/sass/tree/visitors/set_options.rb +125 -0
- data/lib/sass/tree/visitors/to_css.rb +228 -0
- data/lib/sass/tree/warn_node.rb +18 -0
- data/lib/sass/tree/while_node.rb +18 -0
- data/lib/sass/util/multibyte_string_scanner.rb +155 -0
- data/lib/sass/util/subset_map.rb +109 -0
- data/lib/sass/util/test.rb +10 -0
- data/lib/sass/util.rb +948 -0
- data/lib/sass/version.rb +126 -0
- data/lib/sass.rb +95 -0
- data/rails/init.rb +1 -0
- data/test/Gemfile +3 -0
- data/test/Gemfile.lock +10 -0
- data/test/sass/cache_test.rb +89 -0
- data/test/sass/callbacks_test.rb +61 -0
- data/test/sass/conversion_test.rb +1760 -0
- data/test/sass/css2sass_test.rb +458 -0
- data/test/sass/data/hsl-rgb.txt +319 -0
- data/test/sass/engine_test.rb +3244 -0
- data/test/sass/exec_test.rb +86 -0
- data/test/sass/extend_test.rb +1482 -0
- data/test/sass/fixtures/test_staleness_check_across_importers.css +1 -0
- data/test/sass/fixtures/test_staleness_check_across_importers.scss +1 -0
- data/test/sass/functions_test.rb +1139 -0
- data/test/sass/importer_test.rb +192 -0
- data/test/sass/logger_test.rb +58 -0
- data/test/sass/mock_importer.rb +49 -0
- data/test/sass/more_results/more1.css +9 -0
- data/test/sass/more_results/more1_with_line_comments.css +26 -0
- data/test/sass/more_results/more_import.css +29 -0
- data/test/sass/more_templates/_more_partial.sass +2 -0
- data/test/sass/more_templates/more1.sass +23 -0
- data/test/sass/more_templates/more_import.sass +11 -0
- data/test/sass/plugin_test.rb +564 -0
- data/test/sass/results/alt.css +4 -0
- data/test/sass/results/basic.css +9 -0
- data/test/sass/results/cached_import_option.css +3 -0
- data/test/sass/results/compact.css +5 -0
- data/test/sass/results/complex.css +86 -0
- data/test/sass/results/compressed.css +1 -0
- data/test/sass/results/expanded.css +19 -0
- data/test/sass/results/filename_fn.css +3 -0
- data/test/sass/results/if.css +3 -0
- data/test/sass/results/import.css +31 -0
- data/test/sass/results/import_charset.css +5 -0
- data/test/sass/results/import_charset_1_8.css +5 -0
- data/test/sass/results/import_charset_ibm866.css +5 -0
- data/test/sass/results/import_content.css +1 -0
- data/test/sass/results/line_numbers.css +49 -0
- data/test/sass/results/mixins.css +95 -0
- data/test/sass/results/multiline.css +24 -0
- data/test/sass/results/nested.css +22 -0
- data/test/sass/results/options.css +1 -0
- data/test/sass/results/parent_ref.css +13 -0
- data/test/sass/results/script.css +16 -0
- data/test/sass/results/scss_import.css +31 -0
- data/test/sass/results/scss_importee.css +2 -0
- data/test/sass/results/subdir/nested_subdir/nested_subdir.css +1 -0
- data/test/sass/results/subdir/subdir.css +3 -0
- data/test/sass/results/units.css +11 -0
- data/test/sass/results/warn.css +0 -0
- data/test/sass/results/warn_imported.css +0 -0
- data/test/sass/script_conversion_test.rb +299 -0
- data/test/sass/script_test.rb +622 -0
- data/test/sass/scss/css_test.rb +1100 -0
- data/test/sass/scss/rx_test.rb +156 -0
- data/test/sass/scss/scss_test.rb +2106 -0
- data/test/sass/scss/test_helper.rb +37 -0
- data/test/sass/templates/_cached_import_option_partial.scss +1 -0
- data/test/sass/templates/_double_import_loop2.sass +1 -0
- data/test/sass/templates/_filename_fn_import.scss +11 -0
- data/test/sass/templates/_imported_charset_ibm866.sass +4 -0
- data/test/sass/templates/_imported_charset_utf8.sass +4 -0
- data/test/sass/templates/_imported_content.sass +3 -0
- data/test/sass/templates/_partial.sass +2 -0
- data/test/sass/templates/_same_name_different_partiality.scss +1 -0
- data/test/sass/templates/alt.sass +16 -0
- data/test/sass/templates/basic.sass +23 -0
- data/test/sass/templates/bork1.sass +2 -0
- data/test/sass/templates/bork2.sass +2 -0
- data/test/sass/templates/bork3.sass +2 -0
- data/test/sass/templates/bork4.sass +2 -0
- data/test/sass/templates/bork5.sass +3 -0
- data/test/sass/templates/cached_import_option.scss +3 -0
- data/test/sass/templates/compact.sass +17 -0
- data/test/sass/templates/complex.sass +305 -0
- data/test/sass/templates/compressed.sass +15 -0
- data/test/sass/templates/double_import_loop1.sass +1 -0
- data/test/sass/templates/expanded.sass +17 -0
- data/test/sass/templates/filename_fn.scss +18 -0
- data/test/sass/templates/if.sass +11 -0
- data/test/sass/templates/import.sass +12 -0
- data/test/sass/templates/import_charset.sass +9 -0
- data/test/sass/templates/import_charset_1_8.sass +6 -0
- data/test/sass/templates/import_charset_ibm866.sass +11 -0
- data/test/sass/templates/import_content.sass +4 -0
- data/test/sass/templates/importee.less +2 -0
- data/test/sass/templates/importee.sass +19 -0
- data/test/sass/templates/line_numbers.sass +13 -0
- data/test/sass/templates/mixin_bork.sass +5 -0
- data/test/sass/templates/mixins.sass +76 -0
- data/test/sass/templates/multiline.sass +20 -0
- data/test/sass/templates/nested.sass +25 -0
- data/test/sass/templates/nested_bork1.sass +2 -0
- data/test/sass/templates/nested_bork2.sass +2 -0
- data/test/sass/templates/nested_bork3.sass +2 -0
- data/test/sass/templates/nested_bork4.sass +2 -0
- data/test/sass/templates/nested_import.sass +2 -0
- data/test/sass/templates/nested_mixin_bork.sass +6 -0
- data/test/sass/templates/options.sass +2 -0
- data/test/sass/templates/parent_ref.sass +25 -0
- data/test/sass/templates/same_name_different_ext.sass +2 -0
- data/test/sass/templates/same_name_different_ext.scss +1 -0
- data/test/sass/templates/same_name_different_partiality.scss +1 -0
- data/test/sass/templates/script.sass +101 -0
- data/test/sass/templates/scss_import.scss +11 -0
- data/test/sass/templates/scss_importee.scss +1 -0
- data/test/sass/templates/single_import_loop.sass +1 -0
- data/test/sass/templates/subdir/import_up1.scss +1 -0
- data/test/sass/templates/subdir/import_up2.scss +1 -0
- data/test/sass/templates/subdir/nested_subdir/_nested_partial.sass +2 -0
- data/test/sass/templates/subdir/nested_subdir/nested_subdir.sass +3 -0
- data/test/sass/templates/subdir/subdir.sass +6 -0
- data/test/sass/templates/units.sass +11 -0
- data/test/sass/templates/warn.sass +3 -0
- data/test/sass/templates/warn_imported.sass +4 -0
- data/test/sass/test_helper.rb +8 -0
- data/test/sass/util/multibyte_string_scanner_test.rb +147 -0
- data/test/sass/util/subset_map_test.rb +91 -0
- data/test/sass/util_test.rb +382 -0
- data/test/test_helper.rb +80 -0
- metadata +354 -0
@@ -0,0 +1,228 @@
|
|
1
|
+
# A visitor for converting a Sass tree into CSS.
|
2
|
+
class Sass::Tree::Visitors::ToCss < Sass::Tree::Visitors::Base
|
3
|
+
protected
|
4
|
+
|
5
|
+
def initialize
|
6
|
+
@tabs = 0
|
7
|
+
end
|
8
|
+
|
9
|
+
def visit(node)
|
10
|
+
super
|
11
|
+
rescue Sass::SyntaxError => e
|
12
|
+
e.modify_backtrace(:filename => node.filename, :line => node.line)
|
13
|
+
raise e
|
14
|
+
end
|
15
|
+
|
16
|
+
def with_tabs(tabs)
|
17
|
+
old_tabs, @tabs = @tabs, tabs
|
18
|
+
yield
|
19
|
+
ensure
|
20
|
+
@tabs = old_tabs
|
21
|
+
end
|
22
|
+
|
23
|
+
def visit_root(node)
|
24
|
+
result = String.new
|
25
|
+
node.children.each do |child|
|
26
|
+
next if child.invisible?
|
27
|
+
child_str = visit(child)
|
28
|
+
result << child_str + (node.style == :compressed ? '' : "\n")
|
29
|
+
end
|
30
|
+
result.rstrip!
|
31
|
+
return "" if result.empty?
|
32
|
+
result << "\n"
|
33
|
+
unless Sass::Util.ruby1_8? || result.ascii_only?
|
34
|
+
if node.children.first.is_a?(Sass::Tree::CharsetNode)
|
35
|
+
begin
|
36
|
+
encoding = node.children.first.name
|
37
|
+
# Default to big-endian encoding, because we have to decide somehow
|
38
|
+
encoding << 'BE' if encoding =~ /\Autf-(16|32)\Z/i
|
39
|
+
result = result.encode(Encoding.find(encoding))
|
40
|
+
rescue EncodingError
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
result = "@charset \"#{result.encoding.name}\";#{
|
45
|
+
node.style == :compressed ? '' : "\n"
|
46
|
+
}".encode(result.encoding) + result
|
47
|
+
end
|
48
|
+
result
|
49
|
+
rescue Sass::SyntaxError => e
|
50
|
+
e.sass_template ||= node.template
|
51
|
+
raise e
|
52
|
+
end
|
53
|
+
|
54
|
+
def visit_charset(node)
|
55
|
+
"@charset \"#{node.name}\";"
|
56
|
+
end
|
57
|
+
|
58
|
+
def visit_comment(node)
|
59
|
+
return if node.invisible?
|
60
|
+
spaces = (' ' * [@tabs - node.resolved_value[/^ */].size, 0].max)
|
61
|
+
|
62
|
+
content = node.resolved_value.gsub(/^/, spaces)
|
63
|
+
content.gsub!(%r{^(\s*)//(.*)$}) {|md| "#{$1}/*#{$2} */"} if node.type == :silent
|
64
|
+
content.gsub!(/\n +(\* *(?!\/))?/, ' ') if (node.style == :compact || node.style == :compressed) && node.type != :loud
|
65
|
+
content
|
66
|
+
end
|
67
|
+
|
68
|
+
def visit_directive(node)
|
69
|
+
was_in_directive = @in_directive
|
70
|
+
tab_str = ' ' * @tabs
|
71
|
+
return tab_str + node.resolved_value + ";" unless node.has_children
|
72
|
+
return tab_str + node.resolved_value + " {}" if node.children.empty?
|
73
|
+
@in_directive = @in_directive || !node.is_a?(Sass::Tree::MediaNode)
|
74
|
+
result = if node.style == :compressed
|
75
|
+
"#{node.resolved_value}{"
|
76
|
+
else
|
77
|
+
"#{tab_str}#{node.resolved_value} {" + (node.style == :compact ? ' ' : "\n")
|
78
|
+
end
|
79
|
+
was_prop = false
|
80
|
+
first = true
|
81
|
+
node.children.each do |child|
|
82
|
+
next if child.invisible?
|
83
|
+
if node.style == :compact
|
84
|
+
if child.is_a?(Sass::Tree::PropNode)
|
85
|
+
with_tabs(first || was_prop ? 0 : @tabs + 1) {result << visit(child) << ' '}
|
86
|
+
else
|
87
|
+
result[-1] = "\n" if was_prop
|
88
|
+
rendered = with_tabs(@tabs + 1) {visit(child).dup}
|
89
|
+
rendered = rendered.lstrip if first
|
90
|
+
result << rendered.rstrip + "\n"
|
91
|
+
end
|
92
|
+
was_prop = child.is_a?(Sass::Tree::PropNode)
|
93
|
+
first = false
|
94
|
+
elsif node.style == :compressed
|
95
|
+
result << (was_prop ? ";" : "") << with_tabs(0) {visit(child)}
|
96
|
+
was_prop = child.is_a?(Sass::Tree::PropNode)
|
97
|
+
else
|
98
|
+
result << with_tabs(@tabs + 1) {visit(child)} + "\n"
|
99
|
+
end
|
100
|
+
end
|
101
|
+
result.rstrip + if node.style == :compressed
|
102
|
+
"}"
|
103
|
+
else
|
104
|
+
(node.style == :expanded ? "\n" : " ") + "}\n"
|
105
|
+
end
|
106
|
+
ensure
|
107
|
+
@in_directive = was_in_directive
|
108
|
+
end
|
109
|
+
|
110
|
+
def visit_media(node)
|
111
|
+
str = with_tabs(@tabs + node.tabs) {visit_directive(node)}
|
112
|
+
str.gsub!(/\n\Z/, '') unless node.style == :compressed || node.group_end
|
113
|
+
str
|
114
|
+
end
|
115
|
+
|
116
|
+
def visit_supports(node)
|
117
|
+
visit_media(node)
|
118
|
+
end
|
119
|
+
|
120
|
+
def visit_cssimport(node)
|
121
|
+
visit_directive(node)
|
122
|
+
end
|
123
|
+
|
124
|
+
def visit_prop(node)
|
125
|
+
return if node.resolved_value.empty?
|
126
|
+
tab_str = ' ' * (@tabs + node.tabs)
|
127
|
+
if node.style == :compressed
|
128
|
+
"#{tab_str}#{node.resolved_name}:#{node.resolved_value}"
|
129
|
+
else
|
130
|
+
"#{tab_str}#{node.resolved_name}: #{node.resolved_value};"
|
131
|
+
end
|
132
|
+
end
|
133
|
+
|
134
|
+
def visit_rule(node)
|
135
|
+
with_tabs(@tabs + node.tabs) do
|
136
|
+
rule_separator = node.style == :compressed ? ',' : ', '
|
137
|
+
line_separator =
|
138
|
+
case node.style
|
139
|
+
when :nested, :expanded; "\n"
|
140
|
+
when :compressed; ""
|
141
|
+
else; " "
|
142
|
+
end
|
143
|
+
rule_indent = ' ' * @tabs
|
144
|
+
per_rule_indent, total_indent = [:nested, :expanded].include?(node.style) ? [rule_indent, ''] : ['', rule_indent]
|
145
|
+
|
146
|
+
joined_rules = node.resolved_rules.members.map do |seq|
|
147
|
+
next if seq.has_placeholder?
|
148
|
+
rule_part = seq.to_a.join
|
149
|
+
if node.style == :compressed
|
150
|
+
rule_part.gsub!(/([^,])\s*\n\s*/m, '\1 ')
|
151
|
+
rule_part.gsub!(/\s*([,+>])\s*/m, '\1')
|
152
|
+
rule_part.strip!
|
153
|
+
end
|
154
|
+
rule_part
|
155
|
+
end.compact.join(rule_separator)
|
156
|
+
|
157
|
+
joined_rules.sub!(/\A\s*/, per_rule_indent)
|
158
|
+
joined_rules.gsub!(/\s*\n\s*/, "#{line_separator}#{per_rule_indent}")
|
159
|
+
total_rule = total_indent << joined_rules
|
160
|
+
|
161
|
+
to_return = ''
|
162
|
+
old_spaces = ' ' * @tabs
|
163
|
+
if node.style != :compressed
|
164
|
+
if node.options[:debug_info] && !@in_directive
|
165
|
+
to_return << visit(debug_info_rule(node.debug_info, node.options)) << "\n"
|
166
|
+
elsif node.options[:trace_selectors]
|
167
|
+
to_return << "#{old_spaces}/* "
|
168
|
+
to_return << node.stack_trace.join("\n #{old_spaces}")
|
169
|
+
to_return << " */\n"
|
170
|
+
elsif node.options[:line_comments]
|
171
|
+
to_return << "#{old_spaces}/* line #{node.line}"
|
172
|
+
|
173
|
+
if node.filename
|
174
|
+
relative_filename = if node.options[:css_filename]
|
175
|
+
begin
|
176
|
+
Pathname.new(node.filename).relative_path_from(
|
177
|
+
Pathname.new(File.dirname(node.options[:css_filename]))).to_s
|
178
|
+
rescue ArgumentError
|
179
|
+
nil
|
180
|
+
end
|
181
|
+
end
|
182
|
+
relative_filename ||= node.filename
|
183
|
+
to_return << ", #{relative_filename}"
|
184
|
+
end
|
185
|
+
|
186
|
+
to_return << " */\n"
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
if node.style == :compact
|
191
|
+
properties = with_tabs(0) {node.children.map {|a| visit(a)}.join(' ')}
|
192
|
+
to_return << "#{total_rule} { #{properties} }#{"\n" if node.group_end}"
|
193
|
+
elsif node.style == :compressed
|
194
|
+
properties = with_tabs(0) {node.children.map {|a| visit(a)}.join(';')}
|
195
|
+
to_return << "#{total_rule}{#{properties}}"
|
196
|
+
else
|
197
|
+
properties = with_tabs(@tabs + 1) {node.children.map {|a| visit(a)}.join("\n")}
|
198
|
+
end_props = (node.style == :expanded ? "\n" + old_spaces : ' ')
|
199
|
+
to_return << "#{total_rule} {\n#{properties}#{end_props}}#{"\n" if node.group_end}"
|
200
|
+
end
|
201
|
+
|
202
|
+
to_return
|
203
|
+
end
|
204
|
+
end
|
205
|
+
|
206
|
+
private
|
207
|
+
|
208
|
+
def debug_info_rule(debug_info, options)
|
209
|
+
node = Sass::Tree::DirectiveNode.resolved("@media -sass-debug-info")
|
210
|
+
Sass::Util.hash_to_a(debug_info.map {|k, v| [k.to_s, v.to_s]}).each do |k, v|
|
211
|
+
rule = Sass::Tree::RuleNode.new([""])
|
212
|
+
rule.resolved_rules = Sass::Selector::CommaSequence.new(
|
213
|
+
[Sass::Selector::Sequence.new(
|
214
|
+
[Sass::Selector::SimpleSequence.new(
|
215
|
+
[Sass::Selector::Element.new(k.to_s.gsub(/[^\w-]/, "\\\\\\0"), nil)],
|
216
|
+
false)
|
217
|
+
])
|
218
|
+
])
|
219
|
+
prop = Sass::Tree::PropNode.new([""], Sass::Script::String.new(''), :new)
|
220
|
+
prop.resolved_name = "font-family"
|
221
|
+
prop.resolved_value = Sass::SCSS::RX.escape_ident(v.to_s)
|
222
|
+
rule << prop
|
223
|
+
node << rule
|
224
|
+
end
|
225
|
+
node.options = options.merge(:debug_info => false, :line_comments => false, :style => :compressed)
|
226
|
+
node
|
227
|
+
end
|
228
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module Sass
|
2
|
+
module Tree
|
3
|
+
# A dynamic node representing a Sass `@warn` statement.
|
4
|
+
#
|
5
|
+
# @see Sass::Tree
|
6
|
+
class WarnNode < Node
|
7
|
+
# The expression to print.
|
8
|
+
# @return [Script::Node]
|
9
|
+
attr_accessor :expr
|
10
|
+
|
11
|
+
# @param expr [Script::Node] The expression to print
|
12
|
+
def initialize(expr)
|
13
|
+
@expr = expr
|
14
|
+
super()
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
require 'sass/tree/node'
|
2
|
+
|
3
|
+
module Sass::Tree
|
4
|
+
# A dynamic node representing a Sass `@while` loop.
|
5
|
+
#
|
6
|
+
# @see Sass::Tree
|
7
|
+
class WhileNode < Node
|
8
|
+
# The parse tree for the continuation expression.
|
9
|
+
# @return [Script::Node]
|
10
|
+
attr_accessor :expr
|
11
|
+
|
12
|
+
# @param expr [Script::Node] See \{#expr}
|
13
|
+
def initialize(expr)
|
14
|
+
@expr = expr
|
15
|
+
super()
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,155 @@
|
|
1
|
+
require 'strscan'
|
2
|
+
|
3
|
+
if Sass::Util.ruby1_8?
|
4
|
+
Sass::Util::MultibyteStringScanner = StringScanner
|
5
|
+
else
|
6
|
+
if Sass::Util.rbx?
|
7
|
+
# Rubinius's StringScanner class implements some of its methods in terms of
|
8
|
+
# others, which causes us to double-count bytes in some cases if we do
|
9
|
+
# straightforward inheritance. To work around this, we use a delegate class.
|
10
|
+
require 'delegate'
|
11
|
+
class Sass::Util::MultibyteStringScanner < DelegateClass(StringScanner)
|
12
|
+
def initialize(str)
|
13
|
+
super(StringScanner.new(str))
|
14
|
+
@mb_pos = 0
|
15
|
+
@mb_matched_size = nil
|
16
|
+
@mb_last_pos = nil
|
17
|
+
end
|
18
|
+
|
19
|
+
def is_a?(klass)
|
20
|
+
__getobj__.is_a?(klass) || super
|
21
|
+
end
|
22
|
+
end
|
23
|
+
else
|
24
|
+
class Sass::Util::MultibyteStringScanner < StringScanner
|
25
|
+
def initialize(str)
|
26
|
+
super
|
27
|
+
@mb_pos = 0
|
28
|
+
@mb_matched_size = nil
|
29
|
+
@mb_last_pos = nil
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
33
|
+
|
34
|
+
# A wrapper of the native StringScanner class that works correctly with
|
35
|
+
# multibyte character encodings. The native class deals only in bytes, not
|
36
|
+
# characters, for methods like [#pos] and [#matched_size]. This class deals
|
37
|
+
# only in characters, instead.
|
38
|
+
class Sass::Util::MultibyteStringScanner
|
39
|
+
def self.new(str)
|
40
|
+
return StringScanner.new(str) if str.ascii_only?
|
41
|
+
super
|
42
|
+
end
|
43
|
+
|
44
|
+
alias_method :byte_pos, :pos
|
45
|
+
alias_method :byte_matched_size, :matched_size
|
46
|
+
|
47
|
+
def check(pattern); _match super; end
|
48
|
+
def check_until(pattern); _matched super; end
|
49
|
+
def getch; _forward _match super; end
|
50
|
+
def match?(pattern); _size check(pattern); end
|
51
|
+
def matched_size; @mb_matched_size; end
|
52
|
+
def peek(len); string[@mb_pos, len]; end
|
53
|
+
alias_method :peep, :peek
|
54
|
+
def pos; @mb_pos; end
|
55
|
+
alias_method :pointer, :pos
|
56
|
+
def rest_size; rest.size; end
|
57
|
+
def scan(pattern); _forward _match super; end
|
58
|
+
def scan_until(pattern); _forward _matched super; end
|
59
|
+
def skip(pattern); _size scan(pattern); end
|
60
|
+
def skip_until(pattern); _matched _size scan_until(pattern); end
|
61
|
+
|
62
|
+
def get_byte
|
63
|
+
raise "MultibyteStringScanner doesn't support #get_byte."
|
64
|
+
end
|
65
|
+
|
66
|
+
def getbyte
|
67
|
+
raise "MultibyteStringScanner doesn't support #getbyte."
|
68
|
+
end
|
69
|
+
|
70
|
+
def pos=(n)
|
71
|
+
@mb_last_pos = nil
|
72
|
+
|
73
|
+
# We set position kind of a lot during parsing, so we want it to be as
|
74
|
+
# efficient as possible. This is complicated by the fact that UTF-8 is a
|
75
|
+
# variable-length encoding, so it's difficult to find the byte length that
|
76
|
+
# corresponds to a given character length.
|
77
|
+
#
|
78
|
+
# Our heuristic here is to try to count the fewest possible characters. So
|
79
|
+
# if the new position is close to the current one, just count the
|
80
|
+
# characters between the two; if the new position is closer to the
|
81
|
+
# beginning of the string, just count the characters from there.
|
82
|
+
if @mb_pos - n < @mb_pos / 2
|
83
|
+
# New position is close to old position
|
84
|
+
byte_delta = @mb_pos > n ? -string[n...@mb_pos].bytesize : string[@mb_pos...n].bytesize
|
85
|
+
super(byte_pos + byte_delta)
|
86
|
+
else
|
87
|
+
# New position is close to BOS
|
88
|
+
super(string[0...n].bytesize)
|
89
|
+
end
|
90
|
+
@mb_pos = n
|
91
|
+
end
|
92
|
+
|
93
|
+
def reset
|
94
|
+
@mb_pos = 0
|
95
|
+
@mb_matched_size = nil
|
96
|
+
@mb_last_pos = nil
|
97
|
+
super
|
98
|
+
end
|
99
|
+
|
100
|
+
def scan_full(pattern, advance_pointer_p, return_string_p)
|
101
|
+
res = _match super(pattern, advance_pointer_p, true)
|
102
|
+
_forward res if advance_pointer_p
|
103
|
+
return res if return_string_p
|
104
|
+
end
|
105
|
+
|
106
|
+
def search_full(pattern, advance_pointer_p, return_string_p)
|
107
|
+
res = super(pattern, advance_pointer_p, true)
|
108
|
+
_forward res if advance_pointer_p
|
109
|
+
_matched((res if return_string_p))
|
110
|
+
end
|
111
|
+
|
112
|
+
def string=(str)
|
113
|
+
@mb_pos = 0
|
114
|
+
@mb_matched_size = nil
|
115
|
+
@mb_last_pos = nil
|
116
|
+
super
|
117
|
+
end
|
118
|
+
|
119
|
+
def terminate
|
120
|
+
@mb_pos = string.size
|
121
|
+
@mb_matched_size = nil
|
122
|
+
@mb_last_pos = nil
|
123
|
+
super
|
124
|
+
end
|
125
|
+
alias_method :clear, :terminate
|
126
|
+
|
127
|
+
def unscan
|
128
|
+
super
|
129
|
+
@mb_pos = @mb_last_pos
|
130
|
+
@mb_last_pos = @mb_matched_size = nil
|
131
|
+
end
|
132
|
+
|
133
|
+
private
|
134
|
+
|
135
|
+
def _size(str)
|
136
|
+
str && str.size
|
137
|
+
end
|
138
|
+
|
139
|
+
def _match(str)
|
140
|
+
@mb_matched_size = str && str.size
|
141
|
+
str
|
142
|
+
end
|
143
|
+
|
144
|
+
def _matched(res)
|
145
|
+
_match matched
|
146
|
+
res
|
147
|
+
end
|
148
|
+
|
149
|
+
def _forward(str)
|
150
|
+
@mb_last_pos = @mb_pos
|
151
|
+
@mb_pos += str.size if str
|
152
|
+
str
|
153
|
+
end
|
154
|
+
end
|
155
|
+
end
|
@@ -0,0 +1,109 @@
|
|
1
|
+
require 'set'
|
2
|
+
|
3
|
+
module Sass
|
4
|
+
module Util
|
5
|
+
# A map from sets to values.
|
6
|
+
# A value is \{#\[]= set} by providing a set (the "set-set") and a value,
|
7
|
+
# which is then recorded as corresponding to that set.
|
8
|
+
# Values are \{#\[] accessed} by providing a set (the "get-set")
|
9
|
+
# and returning all values that correspond to set-sets
|
10
|
+
# that are subsets of the get-set.
|
11
|
+
#
|
12
|
+
# SubsetMap preserves the order of values as they're inserted.
|
13
|
+
#
|
14
|
+
# @example
|
15
|
+
# ssm = SubsetMap.new
|
16
|
+
# ssm[Set[1, 2]] = "Foo"
|
17
|
+
# ssm[Set[2, 3]] = "Bar"
|
18
|
+
# ssm[Set[1, 2, 3]] = "Baz"
|
19
|
+
#
|
20
|
+
# ssm[Set[1, 2, 3]] #=> ["Foo", "Bar", "Baz"]
|
21
|
+
class SubsetMap
|
22
|
+
# Creates a new, empty SubsetMap.
|
23
|
+
def initialize
|
24
|
+
@hash = {}
|
25
|
+
@vals = []
|
26
|
+
end
|
27
|
+
|
28
|
+
# Whether or not this SubsetMap has any key-value pairs.
|
29
|
+
#
|
30
|
+
# @return [Boolean]
|
31
|
+
def empty?
|
32
|
+
@hash.empty?
|
33
|
+
end
|
34
|
+
|
35
|
+
# Associates a value with a set.
|
36
|
+
# When `set` or any of its supersets is accessed,
|
37
|
+
# `value` will be among the values returned.
|
38
|
+
#
|
39
|
+
# Note that if the same `set` is passed to this method multiple times,
|
40
|
+
# all given `value`s will be associated with that `set`.
|
41
|
+
#
|
42
|
+
# This runs in `O(n)` time, where `n` is the size of `set`.
|
43
|
+
#
|
44
|
+
# @param set [#to_set] The set to use as the map key. May not be empty.
|
45
|
+
# @param value [Object] The value to associate with `set`.
|
46
|
+
# @raise [ArgumentError] If `set` is empty.
|
47
|
+
def []=(set, value)
|
48
|
+
raise ArgumentError.new("SubsetMap keys may not be empty.") if set.empty?
|
49
|
+
|
50
|
+
index = @vals.size
|
51
|
+
@vals << value
|
52
|
+
set.each do |k|
|
53
|
+
@hash[k] ||= []
|
54
|
+
@hash[k] << [set, set.to_set, index]
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
# Returns all values associated with subsets of `set`.
|
59
|
+
#
|
60
|
+
# In the worst case, this runs in `O(m*max(n, log m))` time,
|
61
|
+
# where `n` is the size of `set`
|
62
|
+
# and `m` is the number of assocations in the map.
|
63
|
+
# However, unless many keys in the map overlap with `set`,
|
64
|
+
# `m` will typically be much smaller.
|
65
|
+
#
|
66
|
+
# @param set [Set] The set to use as the map key.
|
67
|
+
# @return [Array<(Object, #to_set)>] An array of pairs,
|
68
|
+
# where the first value is the value associated with a subset of `set`,
|
69
|
+
# and the second value is that subset of `set`
|
70
|
+
# (or whatever `#to_set` object was used to set the value)
|
71
|
+
# This array is in insertion order.
|
72
|
+
# @see #[]
|
73
|
+
def get(set)
|
74
|
+
res = set.map do |k|
|
75
|
+
next unless subsets = @hash[k]
|
76
|
+
subsets.map do |subenum, subset, index|
|
77
|
+
next unless subset.subset?(set)
|
78
|
+
[index, subenum]
|
79
|
+
end
|
80
|
+
end
|
81
|
+
res = Sass::Util.flatten(res, 1)
|
82
|
+
res.compact!
|
83
|
+
res.uniq!
|
84
|
+
res.sort!
|
85
|
+
res.map! {|i, s| [@vals[i], s]}
|
86
|
+
return res
|
87
|
+
end
|
88
|
+
|
89
|
+
# Same as \{#get}, but doesn't return the subsets of the argument
|
90
|
+
# for which values were found.
|
91
|
+
#
|
92
|
+
# @param set [Set] The set to use as the map key.
|
93
|
+
# @return [Array] The array of all values
|
94
|
+
# associated with subsets of `set`, in insertion order.
|
95
|
+
# @see #get
|
96
|
+
def [](set)
|
97
|
+
get(set).map {|v, _| v}
|
98
|
+
end
|
99
|
+
|
100
|
+
# Iterates over each value in the subset map. Ignores keys completely. If
|
101
|
+
# multiple keys have the same value, this will return them multiple times.
|
102
|
+
#
|
103
|
+
# @yield [Object] Each value in the map.
|
104
|
+
def each_value
|
105
|
+
@vals.each {|v| yield v}
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
end
|