csscss 1.1.0 → 1.2.0
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG.md +7 -0
- data/CONTRIBUTORS.md +2 -0
- data/Gemfile +2 -0
- data/Gemfile.lock +11 -1
- data/README.md +10 -1
- data/lib/csscss/cli.rb +60 -25
- data/lib/csscss/parser/common.rb +14 -12
- data/lib/csscss/parser/css.rb +7 -1
- data/lib/csscss/redundancy_analyzer.rb +0 -1
- data/lib/csscss/types.rb +12 -2
- data/lib/csscss/version.rb +1 -1
- data/test/csscss/parser/common_test.rb +10 -6
- data/test/csscss/parser/css_test.rb +33 -0
- data/test/csscss/redundancy_analyzer_test.rb +12 -0
- data/test/csscss/types_test.rb +8 -0
- metadata +3 -3
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,10 @@
|
|
1
|
+
## 1.2.0 - 4/14/2013 ##
|
2
|
+
|
3
|
+
* 0 and 0px are now reconciled as redundancies
|
4
|
+
* Disables color support by default for windows & ruby < 2.0
|
5
|
+
* Fixes bug where unquoted url(data...) isn't parsed correctly
|
6
|
+
* Adds support for LESS files
|
7
|
+
|
1
8
|
## 1.1.0 - 4/12/2013 ##
|
2
9
|
|
3
10
|
* Fixes bug where CLI --no-color wasn't respected
|
data/CONTRIBUTORS.md
CHANGED
data/Gemfile
CHANGED
data/Gemfile.lock
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
csscss (1.
|
4
|
+
csscss (1.2.0)
|
5
5
|
colorize
|
6
6
|
parslet (~> 1.5)
|
7
7
|
|
@@ -12,6 +12,7 @@ GEM
|
|
12
12
|
chunky_png (1.2.7)
|
13
13
|
colorize (0.5.8)
|
14
14
|
columnize (0.3.6)
|
15
|
+
commonjs (0.2.6)
|
15
16
|
compass (0.12.2)
|
16
17
|
chunky_png (~> 1.2)
|
17
18
|
fssm (>= 0.2.7)
|
@@ -23,6 +24,9 @@ GEM
|
|
23
24
|
debugger-linecache (1.2.0)
|
24
25
|
debugger-ruby_core_source (1.2.0)
|
25
26
|
fssm (0.2.10)
|
27
|
+
less (2.3.1)
|
28
|
+
commonjs (~> 0.2.6)
|
29
|
+
libv8 (3.11.8.17)
|
26
30
|
m (1.3.1)
|
27
31
|
method_source (>= 0.6.7)
|
28
32
|
rake (>= 0.9.2.2)
|
@@ -32,8 +36,12 @@ GEM
|
|
32
36
|
parslet (1.5.0)
|
33
37
|
blankslate (~> 2.0)
|
34
38
|
rake (10.0.3)
|
39
|
+
ref (1.0.4)
|
35
40
|
ruby-prof (0.13.0)
|
36
41
|
sass (3.2.7)
|
42
|
+
therubyracer (0.11.4)
|
43
|
+
libv8 (~> 3.11.8.12)
|
44
|
+
ref
|
37
45
|
|
38
46
|
PLATFORMS
|
39
47
|
ruby
|
@@ -42,9 +50,11 @@ DEPENDENCIES
|
|
42
50
|
compass
|
43
51
|
csscss!
|
44
52
|
debugger
|
53
|
+
less
|
45
54
|
m
|
46
55
|
minitest
|
47
56
|
minitest-rg
|
48
57
|
rake
|
49
58
|
ruby-prof
|
50
59
|
sass
|
60
|
+
therubyracer
|
data/README.md
CHANGED
@@ -45,11 +45,20 @@ rulesets that have fewer matches.
|
|
45
45
|
|
46
46
|
$ csscss -n 10 -v path/to/style.css # ignores rulesets with < 10 matches
|
47
47
|
|
48
|
-
If you prefer writing in sass, you can also parse your sass/scss files.
|
48
|
+
If you prefer writing in [sass](http://sass-lang.com/), you can also parse your sass/scss files.
|
49
49
|
|
50
50
|
$ gem install sass
|
51
51
|
$ csscss path/to/style.scss
|
52
52
|
|
53
|
+
If you prefer writing in [LESS](http://lesscss.org/), you can also parse your LESS files.
|
54
|
+
|
55
|
+
$ gem install less
|
56
|
+
$ csscss path/to/style.less
|
57
|
+
|
58
|
+
LESS requires an additional javascript runtime.
|
59
|
+
[v8/therubyracer](https://rubygems.org/gems/therubyracer) on most
|
60
|
+
rubies, and [therubyrhino](https://rubygems.org/gems/therubyrhino) on
|
61
|
+
jruby.
|
53
62
|
|
54
63
|
## I found bugs ##
|
55
64
|
|
data/lib/csscss/cli.rb
CHANGED
@@ -3,12 +3,12 @@ module Csscss
|
|
3
3
|
def initialize(argv)
|
4
4
|
@argv = argv
|
5
5
|
@verbose = false
|
6
|
-
@color =
|
6
|
+
@color = !windows_1_9
|
7
7
|
@minimum = 3
|
8
8
|
@compass = false
|
9
9
|
@ignored_properties = []
|
10
10
|
@ignored_selectors = []
|
11
|
-
@match_shorthand
|
11
|
+
@match_shorthand = true
|
12
12
|
end
|
13
13
|
|
14
14
|
def run
|
@@ -16,31 +16,22 @@ module Csscss
|
|
16
16
|
execute
|
17
17
|
end
|
18
18
|
|
19
|
+
private
|
19
20
|
def execute
|
20
21
|
warn_old_debug_flag if ENV["CSSCSS_DEBUG"]
|
21
22
|
|
22
|
-
all_contents
|
23
|
-
if
|
24
|
-
|
25
|
-
require "sass"
|
26
|
-
rescue LoadError
|
27
|
-
abort "Must install sass gem before parsing sass/scss files"
|
28
|
-
end
|
29
|
-
|
30
|
-
sass_options = {cache:false}
|
31
|
-
sass_options[:load_paths] = Compass.configuration.sass_load_paths if @compass
|
32
|
-
begin
|
33
|
-
Sass::Engine.for_file(filename, sass_options).render
|
34
|
-
rescue Sass::SyntaxError => e
|
35
|
-
if e.message =~ /compass/ && !@compass
|
36
|
-
puts "Enable --compass option to use compass's extensions"
|
37
|
-
exit 1
|
38
|
-
else
|
39
|
-
raise e
|
40
|
-
end
|
41
|
-
end
|
23
|
+
all_contents= @argv.map do |filename|
|
24
|
+
if filename =~ URI.regexp
|
25
|
+
load_css_file(filename)
|
42
26
|
else
|
43
|
-
|
27
|
+
case File.extname(filename).downcase
|
28
|
+
when ".scss", ".sass"
|
29
|
+
load_sass_file(filename)
|
30
|
+
when ".less"
|
31
|
+
load_less_file(filename)
|
32
|
+
else
|
33
|
+
load_css_file(filename)
|
34
|
+
end
|
44
35
|
end
|
45
36
|
end.join("\n")
|
46
37
|
|
@@ -80,7 +71,7 @@ module Csscss
|
|
80
71
|
@verbose = v
|
81
72
|
end
|
82
73
|
|
83
|
-
opts.on("--[no-]color", "Colorize output (default is
|
74
|
+
opts.on("--[no-]color", "Colorize output (default is #{@color})") do |c|
|
84
75
|
@color = c
|
85
76
|
end
|
86
77
|
|
@@ -133,7 +124,6 @@ module Csscss
|
|
133
124
|
print_help(opts)
|
134
125
|
end
|
135
126
|
|
136
|
-
private
|
137
127
|
def print_help(opts)
|
138
128
|
puts opts
|
139
129
|
exit
|
@@ -155,6 +145,51 @@ module Csscss
|
|
155
145
|
abort "Must install compass gem before enabling its extensions"
|
156
146
|
end
|
157
147
|
|
148
|
+
def windows_1_9
|
149
|
+
RbConfig::CONFIG['host_os'] =~ /mswin|mingw|cygwin/ && RUBY_VERSION =~ /^1\.9/
|
150
|
+
end
|
151
|
+
|
152
|
+
def gem_installed?(gem_name)
|
153
|
+
begin
|
154
|
+
require gem_name
|
155
|
+
true
|
156
|
+
rescue LoadError
|
157
|
+
false
|
158
|
+
end
|
159
|
+
end
|
160
|
+
|
161
|
+
def load_sass_file(filename)
|
162
|
+
if !gem_installed?("sass") then
|
163
|
+
abort 'Must install the "sass" gem before parsing sass/scss files'
|
164
|
+
end
|
165
|
+
|
166
|
+
sass_options = {cache:false}
|
167
|
+
sass_options[:load_paths] = Compass.configuration.sass_load_paths if @compass
|
168
|
+
begin
|
169
|
+
Sass::Engine.for_file(filename, sass_options).render
|
170
|
+
rescue Sass::SyntaxError => e
|
171
|
+
if e.message =~ /compass/ && !@compass
|
172
|
+
puts "Enable --compass option to use compass's extensions"
|
173
|
+
exit 1
|
174
|
+
else
|
175
|
+
raise e
|
176
|
+
end
|
177
|
+
end
|
178
|
+
end
|
179
|
+
|
180
|
+
def load_less_file(filename)
|
181
|
+
if !gem_installed?("less") then
|
182
|
+
abort 'Must install the "less" gem before parsing less files'
|
183
|
+
end
|
184
|
+
|
185
|
+
contents = load_css_file(filename)
|
186
|
+
Less::Parser.new.parse(contents).to_css
|
187
|
+
end
|
188
|
+
|
189
|
+
def load_css_file(filename)
|
190
|
+
open(filename) {|f| f.read }
|
191
|
+
end
|
192
|
+
|
158
193
|
class << self
|
159
194
|
def run(argv)
|
160
195
|
new(argv).run
|
data/lib/csscss/parser/common.rb
CHANGED
@@ -5,17 +5,19 @@ module Csscss
|
|
5
5
|
|
6
6
|
UNITS = %w(px em ex in cm mm pt pc)
|
7
7
|
|
8
|
-
rule(:space)
|
9
|
-
rule(:space?)
|
10
|
-
rule(:number)
|
11
|
-
rule(:numbers)
|
12
|
-
rule(:decimal)
|
13
|
-
rule(:percent)
|
14
|
-
rule(:
|
15
|
-
rule(:
|
16
|
-
rule(:
|
17
|
-
rule(:
|
18
|
-
rule(:
|
8
|
+
rule(:space) { match['\s'].repeat(1) }
|
9
|
+
rule(:space?) { space.maybe }
|
10
|
+
rule(:number) { match["0-9"] }
|
11
|
+
rule(:numbers) { number.repeat(1) }
|
12
|
+
rule(:decimal) { numbers >> str(".").maybe >> numbers.maybe }
|
13
|
+
rule(:percent) { decimal >> stri("%") >> space? }
|
14
|
+
rule(:non_zero_length) { decimal >> stri_list(UNITS) >> space? }
|
15
|
+
rule(:zero_length) { match["0"] }
|
16
|
+
rule(:length) { zero_length | non_zero_length }
|
17
|
+
rule(:identifier) { match["a-zA-Z"].repeat(1) }
|
18
|
+
rule(:inherit) { stri("inherit") }
|
19
|
+
rule(:eof) { any.absent? }
|
20
|
+
rule(:nada) { any.repeat.as(:nada) }
|
19
21
|
|
20
22
|
rule(:http) {
|
21
23
|
(match['a-zA-Z0-9.:/\-'] | str('\(') | str('\)')).repeat >> space?
|
@@ -29,7 +31,7 @@ module Csscss
|
|
29
31
|
stri("url") >> parens do
|
30
32
|
(any_quoted { http } >> space?) |
|
31
33
|
(any_quoted { data } >> space?) |
|
32
|
-
http
|
34
|
+
data | http
|
33
35
|
end
|
34
36
|
}
|
35
37
|
|
data/lib/csscss/parser/css.rb
CHANGED
@@ -25,7 +25,13 @@ module Csscss
|
|
25
25
|
rule(:attribute) {
|
26
26
|
match["^:{}"].repeat(1).as(:property) >>
|
27
27
|
str(":") >>
|
28
|
-
match["^;}"].repeat(1).
|
28
|
+
(match["^;}"].repeat(1).capture(:stuff) >> dynamic {|source, context|
|
29
|
+
if context.captures[:stuff].to_s =~ /data:/
|
30
|
+
str(";") >> match["^;}"].repeat(1)
|
31
|
+
else
|
32
|
+
any.present?
|
33
|
+
end
|
34
|
+
}).as(:value) >>
|
29
35
|
str(";").maybe >>
|
30
36
|
space?
|
31
37
|
}
|
data/lib/csscss/types.rb
CHANGED
@@ -30,14 +30,15 @@ module Csscss
|
|
30
30
|
|
31
31
|
def ==(other)
|
32
32
|
if other.respond_to?(:property) && other.respond_to?(:value)
|
33
|
-
|
33
|
+
# using eql? tanks performance
|
34
|
+
property == other.property && normalize_value(value) == normalize_value(other.value)
|
34
35
|
else
|
35
36
|
false
|
36
37
|
end
|
37
38
|
end
|
38
39
|
|
39
40
|
def hash
|
40
|
-
[property, value].hash
|
41
|
+
[property, normalize_value(value)].hash
|
41
42
|
end
|
42
43
|
|
43
44
|
def eql?(other)
|
@@ -67,6 +68,15 @@ module Csscss
|
|
67
68
|
"<#{self.class} #{to_s}>"
|
68
69
|
end
|
69
70
|
end
|
71
|
+
|
72
|
+
private
|
73
|
+
def normalize_value(value)
|
74
|
+
if value =~ /^0(#{Csscss::Parser::Common::UNITS.join("|")}|%)$/
|
75
|
+
"0"
|
76
|
+
else
|
77
|
+
value
|
78
|
+
end
|
79
|
+
end
|
70
80
|
end
|
71
81
|
|
72
82
|
class Selector < Struct.new(:selectors)
|
data/lib/csscss/version.rb
CHANGED
@@ -106,6 +106,8 @@ module Csscss::Parser
|
|
106
106
|
@parser.length.must_parse "123px"
|
107
107
|
@parser.length.must_parse "123EM"
|
108
108
|
@parser.length.must_parse "1.23Pt"
|
109
|
+
@parser.length.must_parse "0"
|
110
|
+
@parser.length.wont_parse "1"
|
109
111
|
end
|
110
112
|
end
|
111
113
|
|
@@ -115,15 +117,16 @@ module Csscss::Parser
|
|
115
117
|
@parser.http.must_parse 'foo\(bar\).jpg'
|
116
118
|
@parser.http.must_parse 'http://foo\(bar\).jpg'
|
117
119
|
@parser.http.must_parse 'http://foo.com/baz/\(bar\).jpg'
|
118
|
-
@parser.http.must_parse
|
119
|
-
@parser.http.must_parse
|
120
|
-
@parser.http.must_parse
|
121
|
-
@parser.http.must_parse
|
120
|
+
@parser.http.must_parse "//foo.com/foo.jpg"
|
121
|
+
@parser.http.must_parse "https://foo.com/foo.jpg"
|
122
|
+
@parser.http.must_parse "http://foo100.com/foo.jpg"
|
123
|
+
@parser.http.must_parse "http://foo-bar.com/foo.jpg"
|
122
124
|
end
|
123
125
|
|
124
126
|
it "parses data" do
|
125
|
-
@parser.data.must_parse
|
126
|
-
@parser.data.must_parse
|
127
|
+
@parser.data.must_parse ""
|
128
|
+
@parser.data.must_parse ""
|
129
|
+
@parser.data.must_parse ""
|
127
130
|
end
|
128
131
|
|
129
132
|
it "parses urls" do
|
@@ -133,6 +136,7 @@ module Csscss::Parser
|
|
133
136
|
@parser.url.must_parse "url('foo.jpg')"
|
134
137
|
@parser.url.must_parse "url('foo.jpg' )"
|
135
138
|
@parser.url.must_parse 'url(foo\(bar\).jpg)'
|
139
|
+
@parser.url.must_parse "url()"
|
136
140
|
@parser.url.must_parse "url('')"
|
137
141
|
end
|
138
142
|
end
|
@@ -134,6 +134,39 @@ module Csscss::Parser
|
|
134
134
|
rs(sel("h1"), [dec("display", "none")])
|
135
135
|
])
|
136
136
|
end
|
137
|
+
|
138
|
+
it "parses attributes with encoded data that include semicolons" do
|
139
|
+
trans(%$
|
140
|
+
.foo1 {
|
141
|
+
background: rgb(123, 123, 123) url() repeat-x;
|
142
|
+
display: block;
|
143
|
+
}
|
144
|
+
|
145
|
+
.foo2 {
|
146
|
+
background: white url() repeat-x
|
147
|
+
}
|
148
|
+
|
149
|
+
.foo3 {
|
150
|
+
outline: 1px;
|
151
|
+
background: white url() repeat-x;
|
152
|
+
display: block;
|
153
|
+
}
|
154
|
+
|
155
|
+
.foo4 {
|
156
|
+
background: blue url(images/bg-bolt-inactive.png) no-repeat 99% 5px;
|
157
|
+
display: block;
|
158
|
+
}
|
159
|
+
$).must_equal([
|
160
|
+
rs(sel(".foo1"), [dec("background", "rgb(123, 123, 123) url() repeat-x"),
|
161
|
+
dec("display", "block")]),
|
162
|
+
rs(sel(".foo2"), [dec("background", "white url() repeat-x")]),
|
163
|
+
rs(sel(".foo3"), [dec("outline", "1px"),
|
164
|
+
dec("background", "white url() repeat-x"),
|
165
|
+
dec("display", "block")]),
|
166
|
+
rs(sel(".foo4"), [dec("background", "blue url(images/bg-bolt-inactive.png) no-repeat 99% 5px"),
|
167
|
+
dec("display", "block")])
|
168
|
+
])
|
169
|
+
end
|
137
170
|
end
|
138
171
|
end
|
139
172
|
end
|
@@ -275,6 +275,18 @@ module Csscss
|
|
275
275
|
})
|
276
276
|
end
|
277
277
|
|
278
|
+
it "matches 0 and 0px" do
|
279
|
+
css = %$
|
280
|
+
.bar { padding: 0; }
|
281
|
+
.foo { padding: 0px; }
|
282
|
+
$
|
283
|
+
|
284
|
+
RedundancyAnalyzer.new(css).redundancies.must_equal({
|
285
|
+
[sel(".bar"), sel(".foo")] => [dec("padding", "0")]
|
286
|
+
})
|
287
|
+
end
|
288
|
+
|
289
|
+
|
278
290
|
# TODO: someday
|
279
291
|
# it "reports duplication within the same selector" do
|
280
292
|
# css = %$
|
data/test/csscss/types_test.rb
CHANGED
@@ -69,5 +69,13 @@ module Csscss
|
|
69
69
|
h[dec2].must_equal true
|
70
70
|
h[dec3].must_equal true
|
71
71
|
end
|
72
|
+
|
73
|
+
it "equates 0 length with and without units" do
|
74
|
+
Declaration.new("padding", "0px").must_equal Declaration.new("padding", "0")
|
75
|
+
Declaration.new("padding", "0%").must_equal Declaration.new("padding", "0")
|
76
|
+
Declaration.new("padding", "0").must_equal Declaration.new("padding", "0em")
|
77
|
+
|
78
|
+
Declaration.new("padding", "1").wont_equal Declaration.new("padding", "1px")
|
79
|
+
end
|
72
80
|
end
|
73
81
|
end
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: csscss
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.
|
4
|
+
version: 1.2.0
|
5
5
|
prerelease:
|
6
6
|
platform: ruby
|
7
7
|
authors:
|
@@ -9,7 +9,7 @@ authors:
|
|
9
9
|
autorequire:
|
10
10
|
bindir: bin
|
11
11
|
cert_chain: []
|
12
|
-
date: 2013-04-
|
12
|
+
date: 2013-04-14 00:00:00.000000000 Z
|
13
13
|
dependencies:
|
14
14
|
- !ruby/object:Gem::Dependency
|
15
15
|
name: parslet
|
@@ -127,7 +127,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
127
127
|
version: '0'
|
128
128
|
segments:
|
129
129
|
- 0
|
130
|
-
hash:
|
130
|
+
hash: 4177886489285701178
|
131
131
|
requirements: []
|
132
132
|
rubyforge_project:
|
133
133
|
rubygems_version: 1.8.25
|