XPath compiler support for id()
This has been largely ported over from the Evaluator implementation.
This commit is contained in:
parent
2ff1f9ab4f
commit
7bcd462e22
|
@ -73,16 +73,18 @@ module Oga
|
||||||
vars = variables_literal.assign(self.nil)
|
vars = variables_literal.assign(self.nil)
|
||||||
|
|
||||||
proc_ast = literal(:lambda).add_block(document, vars) do
|
proc_ast = literal(:lambda).add_block(document, vars) do
|
||||||
if return_nodeset?(ast)
|
input_assign = original_input_literal.assign(document)
|
||||||
input_assign = original_input_literal.assign(document)
|
|
||||||
|
|
||||||
matched.assign(literal(XML::NodeSet).new)
|
if return_nodeset?(ast)
|
||||||
|
body = matched.assign(literal(XML::NodeSet).new)
|
||||||
.followed_by(input_assign)
|
.followed_by(input_assign)
|
||||||
.followed_by(ruby_ast)
|
.followed_by(ruby_ast)
|
||||||
.followed_by(matched)
|
.followed_by(matched)
|
||||||
else
|
else
|
||||||
ruby_ast
|
body = ruby_ast
|
||||||
end
|
end
|
||||||
|
|
||||||
|
input_assign.followed_by(body)
|
||||||
end
|
end
|
||||||
|
|
||||||
generator = Ruby::Generator.new
|
generator = Ruby::Generator.new
|
||||||
|
@ -794,6 +796,68 @@ module Oga
|
||||||
.followed_by(count)
|
.followed_by(count)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
##
|
||||||
|
# Processes the `id()` function call.
|
||||||
|
#
|
||||||
|
# The XPath specification states that this function's behaviour should be
|
||||||
|
# controlled by a DTD. If a DTD were to specify that the ID attribute for
|
||||||
|
# a certain element would be "foo" then this function should use said
|
||||||
|
# attribute.
|
||||||
|
#
|
||||||
|
# Oga does not support DTD parsing/evaluation and as such always uses the
|
||||||
|
# "id" attribute.
|
||||||
|
#
|
||||||
|
# This function searches the entire document for a matching node,
|
||||||
|
# regardless of the current position.
|
||||||
|
#
|
||||||
|
# @param [Oga::Ruby::Node] input
|
||||||
|
# @param [AST::Node] arg
|
||||||
|
# @return [Oga::Ruby::Node]
|
||||||
|
#
|
||||||
|
def on_call_id(input, arg)
|
||||||
|
orig_input = original_input_literal
|
||||||
|
node = node_literal
|
||||||
|
ids_var = unique_literal('ids')
|
||||||
|
matched = unique_literal('id_matched')
|
||||||
|
id_str_var = unique_literal('id_string')
|
||||||
|
attr_var = unique_literal('attr')
|
||||||
|
|
||||||
|
matched_assign = matched.assign(literal(XML::NodeSet).new)
|
||||||
|
|
||||||
|
# When using some sort of path we'll want the text of all matched nodes.
|
||||||
|
if return_nodeset?(arg)
|
||||||
|
id_assign = ids_var.assign(literal(:[]))
|
||||||
|
.followed_by(process(arg, input) { |node| ids_var << node.text })
|
||||||
|
|
||||||
|
# For everything else we'll cast the value to a string and split it on
|
||||||
|
# every space.
|
||||||
|
else
|
||||||
|
conversion = literal(Conversion).to_string(ids_var).split(string(' '))
|
||||||
|
|
||||||
|
id_assign = ids_var.assign(process(arg, input))
|
||||||
|
.followed_by(ids_var.assign(conversion))
|
||||||
|
end
|
||||||
|
|
||||||
|
id_str_assign = id_str_var.assign(string('id'))
|
||||||
|
|
||||||
|
each_node = orig_input.each_node.add_block(node) do
|
||||||
|
node.is_a?(XML::Element).if_true do
|
||||||
|
assign = attr_var.assign(node.attribute(id_str_var))
|
||||||
|
|
||||||
|
compare = attr_var.and(ids_var.include?(attr_var.value)).if_true do
|
||||||
|
matched << node
|
||||||
|
end
|
||||||
|
|
||||||
|
assign.followed_by(compare)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
matched_assign.followed_by(id_assign)
|
||||||
|
.followed_by(id_str_assign)
|
||||||
|
.followed_by(each_node)
|
||||||
|
.followed_by(matched)
|
||||||
|
end
|
||||||
|
|
||||||
##
|
##
|
||||||
# Delegates type tests to specific handlers.
|
# Delegates type tests to specific handlers.
|
||||||
#
|
#
|
||||||
|
|
Loading…
Reference in New Issue