oga/lib/oga/css/parser.y

223 lines
4.0 KiB
Plaintext
Raw Normal View History

##
# AST parser for CSS expressions.
#
class Oga::CSS::Parser
token T_IDENT T_PIPE T_LBRACK T_RBRACK T_COLON T_SPACE T_LPAREN T_RPAREN T_MINUS
token T_EQ T_SPACE_IN T_STARTS_WITH T_ENDS_WITH T_IN T_HYPHEN_IN
token T_CHILD T_FOLLOWING T_FOLLOWING_DIRECT
token T_NTH T_INT T_STRING T_ODD T_EVEN T_DOT T_HASH
options no_result_var
prechigh
left T_COLON
left T_CHILD T_FOLLOWING T_FOLLOWING_DIRECT
left T_DOT T_HASH
preclow
rule
css
: expression { val[0] }
| /* none */ { nil }
;
expression
: path
| path_member
;
path
: path_members { s(:path, *val[0]) }
;
path_members
: path_member T_SPACE path_member { [val[0], val[2]] }
| path_member T_SPACE path_members { [val[0], *val[2]] }
;
path_member
: node_test
| axis
| pseudo_class
| class
| id
;
node_test
: node_name { s(:test, *val[0]) }
| node_name predicate { s(:test, *val[0], val[1]) }
;
node_name
# foo
: T_IDENT { [nil, val[0]] }
# ns|foo
| T_IDENT T_PIPE T_IDENT { [val[0], val[2]] }
# |foo
| T_PIPE T_IDENT { [nil, val[1]] }
;
predicate
2014-10-05 08:06:51 +00:00
: T_LBRACK predicate_members T_RBRACK { val[1] }
;
predicate_members
2014-10-05 21:33:46 +00:00
: node_test
| operator
2014-10-05 08:06:51 +00:00
;
class
: class_name { s(:class, val[0]) }
| path_member class_name { s(:class, val[1], val[0]) }
;
class_name
: T_DOT T_IDENT { val[1] }
;
id
: id_name { s(:id, val[0]) }
| path_member id_name { s(:id, val[1], val[0]) }
;
id_name
: T_HASH T_IDENT { val[1] }
;
2014-10-05 08:06:51 +00:00
operator
: op_members T_EQ op_members { s(:eq, val[0], val[2]) }
| op_members T_SPACE_IN op_members { s(:space_in, val[0], val[2]) }
| op_members T_STARTS_WITH op_members { s(:starts_with, val[0], val[2]) }
| op_members T_ENDS_WITH op_members { s(:ends_with, val[0], val[2]) }
| op_members T_IN op_members { s(:in, val[0], val[2]) }
| op_members T_HYPHEN_IN op_members { s(:hyphen_in, val[0],val[2]) }
;
op_members
: node_test
| string
;
2014-10-05 21:33:46 +00:00
axis
# x > y
: path_member T_CHILD path_member
2014-10-05 21:33:46 +00:00
{
s(:axis, 'child', val[0], val[2])
}
# x + y
| path_member T_FOLLOWING path_member
2014-10-05 21:33:46 +00:00
{
s(:axis, 'following', val[0], val[2])
}
# x ~ y
| path_member T_FOLLOWING_DIRECT path_member
2014-10-05 08:06:51 +00:00
{
2014-10-05 21:33:46 +00:00
s(:axis, 'following-direct', val[0], val[2])
2014-10-05 08:06:51 +00:00
}
;
pseudo_class
# :root
: pseudo_name { s(:pseudo, val[0]) }
# x:root
| path_member pseudo_name { s(:pseudo, val[1], val[0]) }
# :nth-child(2)
| pseudo_name pseudo_args { s(:pseudo, val[0], nil, val[1]) }
# x:nth-child(2)
| path_member pseudo_name pseudo_args { s(:pseudo, val[1], val[0], val[2]) }
;
pseudo_name
: T_COLON T_IDENT { val[1] }
;
pseudo_args
: T_LPAREN pseudo_arg T_RPAREN { val[1] }
;
pseudo_arg
: integer
| odd
| even
| nth
;
odd
: T_ODD { s(:odd) }
;
even
: T_EVEN { s(:even) }
;
nth
: T_NTH { s(:nth) }
| T_MINUS T_NTH { s(:nth) }
| integer T_NTH { s(:nth, val[0]) }
| integer T_NTH integer { s(:nth, val[0], val[2]) }
2014-10-05 08:06:51 +00:00
;
string
: T_STRING { s(:string, val[0]) }
;
integer
: T_INT { s(:int, val[0].to_i) }
;
end
---- inner
##
# @param [String] data The input to parse.
#
def initialize(data)
@lexer = Lexer.new(data)
end
##
# @param [Symbol] type
# @param [Array] children
# @return [AST::Node]
#
def s(type, *children)
return AST::Node.new(type, children)
end
##
# Yields the next token from the lexer.
#
# @yieldparam [Array]
#
def yield_next_token
@lexer.advance do |*args|
yield args
end
yield [false, false]
end
##
# Parses the input and returns the corresponding AST.
#
# @example
# parser = Oga::CSS::Parser.new('foo.bar')
# ast = parser.parse
#
# @return [AST::Node]
#
def parse
ast = yyparse(self, :yield_next_token)
return ast
end
# vim: set ft=racc: