NAME
Mojo::DOM - Minimalistic HTML/XML DOM parser with CSS selectors
SYNOPSIS
my $dom = Mojo::DOM->new( '<div><p id="a">Test</p><p id="b">123</p></div>' );
say $dom ->at( '#b' )->text;
say $dom ->find( 'p' )-> map ( 'text' )-> join ( "\n" );
say $dom ->find( '[id]' )-> map ( attr => 'id' )-> join ( "\n" );
$dom ->find( 'p[id]' )-> reverse -> each ( sub { say $_ ->{id} });
for my $e ( $dom ->find( 'p[id]' )-> each ) {
say $e ->{id}, ':' , $e ->text;
}
$dom ->find( 'div p' )-> last ->append( '<p id="c">456</p>' );
$dom ->at( '#c' )->prepend( $dom ->new_tag( 'p' , id => 'd' , '789' ));
$dom ->find( ':not(p)' )-> map ( 'strip' );
say "$dom" ;
|
DESCRIPTION
Mojo::DOM is a minimalistic and relaxed HTML/XML DOM parser with CSS selector support. It will even try to interpret broken HTML and XML, so you should not use it for validation.
NODES AND ELEMENTS
When we parse an HTML/XML fragment, it gets turned into a tree of nodes.
<!DOCTYPE html>
<html>
<head><title>Hello</title></head>
<body>World!</body>
</html>
|
There are currently eight different kinds of nodes, cdata
, comment
, doctype
, pi
, raw
, root
, tag
and text
. Elements are nodes of the type tag
.
root
|- doctype (html)
+- tag (html)
|- tag (head)
| +- tag (title)
| +- raw (Hello)
+- tag (body)
+- text (World!)
|
While all node types are represented as Mojo::DOM objects, some methods like "attr" and "namespace" only apply to elements.
HTML AND XML
Mojo::DOM defaults to HTML semantics, that means all tags and attribute names are lowercased and selectors need to be lowercase as well.
my $dom = Mojo::DOM->new( '<P ID="greeting">Hi!</P>' );
say $dom ->at( 'p[id]' )->text;
|
If an XML declaration is found, the parser will automatically switch into XML mode and everything becomes case-sensitive.
my $dom = Mojo::DOM->new( '<?xml version="1.0"?><P ID="greeting">Hi!</P>' );
say $dom ->at( 'P[ID]' )->text;
|
HTML or XML semantics can also be forced with the "xml" method.
my $dom = Mojo::DOM->new->xml(0)->parse( '<P ID="greeting">Hi!</P>' );
say $dom ->at( 'p[id]' )->text;
my $dom = Mojo::DOM->new->xml(1)->parse( '<P ID="greeting">Hi!</P>' );
say $dom ->at( 'P[ID]' )->text;
|
METHODS
Mojo::DOM implements the following methods.
all_text
my $text = $dom ->all_text;
|
Extract text content from all descendant nodes of this element. For HTML documents script
and style
elements are excluded.
$dom ->parse( "<div>foo\n<p>bar</p>baz\n</div>" )->at( 'div' )->all_text;
|
ancestors
my $collection = $dom ->ancestors;
my $collection = $dom ->ancestors( 'div ~ p' );
|
Find all ancestor elements of this node matching the CSS selector and return a Mojo::Collection object containing these elements as Mojo::DOM objects. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
say $dom ->ancestors-> map ( 'tag' )-> join ( "\n" );
|
append
$dom = $dom ->append( '<p>I ♥ Mojolicious!</p>' );
$dom = $dom ->append(Mojo::DOM->new);
|
Append HTML/XML fragment to this node (for all node types other than root
).
$dom ->parse( '<div><h1>Test</h1></div>' )
->at( 'h1' )->append( '<h2>123</h2>' )->root;
$dom ->parse( '<p>Test</p>' )->at( 'p' )
->child_nodes->first->append( ' 123' )->root;
|
append_content
$dom = $dom ->append_content( '<p>I ♥ Mojolicious!</p>' );
$dom = $dom ->append_content(Mojo::DOM->new);
|
Append HTML/XML fragment (for root
and tag
nodes) or raw content to this node's content.
$dom ->parse( '<div><h1>Test</h1></div>' )
->at( 'h1' )->append_content( '123' )->root;
$dom ->parse( '<!-- Test --><br>' )
->child_nodes->first->append_content( '123 ' )->root;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->append_content( '<i>123</i>' )->root;
|
at
my $result = $dom ->at( 'div ~ p' );
|
Find first descendant element of this element matching the CSS selector and return it as a Mojo::DOM object, or undef
if none could be found. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
my $namespace = $dom ->at( '[xmlns\:svg]' )->{ 'xmlns:svg' };
|
Trailing key/value pairs can be used to declare xml namespace aliases.
attr
my $hash = $dom ->attr;
my $foo = $dom ->attr( 'foo' );
$dom = $dom ->attr({ foo => 'bar' });
$dom = $dom ->attr( foo => 'bar' );
|
This element's attributes.
delete $dom ->attr->{id};
$dom ->attr( selected => undef );
say $dom ->find( '*' )-> map ( attr => 'id' )->compact-> join ( "\n" );
|
child_nodes
my $collection = $dom ->child_nodes;
|
Return a Mojo::Collection object containing all child nodes of this element as Mojo::DOM objects.
$dom ->parse( '<p>Test<b>123</b></p>' )->at( 'p' )->child_nodes->first->remove;
$dom ->parse( '<!DOCTYPE html><b>123</b>' )->child_nodes->first;
$dom ->parse( '<b>123</b><!-- Test -->' )->child_nodes-> last ->content;
|
children
my $collection = $dom ->children;
my $collection = $dom ->children( 'div ~ p' );
|
Find all child elements of this element matching the CSS selector and return a Mojo::Collection object containing these elements as Mojo::DOM objects. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
say $dom ->children->shuffle->first->tag;
|
content
my $str = $dom ->content;
$dom = $dom ->content( '<p>I ♥ Mojolicious!</p>' );
$dom = $dom ->content(Mojo::DOM->new);
|
Return this node's content or replace it with HTML/XML fragment (for root
and tag
nodes) or raw content.
$dom ->parse( '<div><b>Test</b></div>' )->at( 'div' )->content;
$dom ->parse( '<div><h1>Test</h1></div>' )->at( 'h1' )->content( '123' )->root;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->content( '<i>123</i>' )->root;
$dom ->parse( '<div><h1>Test</h1></div>' )->at( 'h1' )->content( '' )->root;
$dom ->parse( '<!-- Test --><br>' )->child_nodes->first->content;
$dom ->parse( '<div><!-- Test -->456</div>' )
->at( 'div' )->child_nodes->first->content( ' 123 ' )->root;
|
descendant_nodes
my $collection = $dom ->descendant_nodes;
|
Return a Mojo::Collection object containing all descendant nodes of this element as Mojo::DOM objects.
$dom ->parse( '<p><!-- Test --><b>123<!-- 456 --></b></p>' )
->descendant_nodes-> grep ( sub { $_ ->type eq 'comment' })
-> map ( 'remove' )->first;
$dom ->parse( '<p><b>123</b>456</p>' )
->at( 'p' )->descendant_nodes-> grep ( sub { $_ ->type eq 'text' })
-> map ( content => 'test' )->first->root;
|
find
my $collection = $dom ->find( 'div ~ p' );
|
Find all descendant elements of this element matching the CSS selector and return a Mojo::Collection object containing these elements as Mojo::DOM objects. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
my $id = $dom ->find( 'div' )->[23]{id};
my @headers = $dom ->find( 'h1, h2, h3' )-> map ( 'text' )-> each ;
my $hash = $dom ->find( '*' )->reduce( sub { $a ->{ $b ->tag}++; $a }, {});
my @divs = $dom ->find( 'div.foo\.bar' )-> each ;
|
Trailing key/value pairs can be used to declare xml namespace aliases.
following
my $collection = $dom ->following;
my $collection = $dom ->following( 'div ~ p' );
|
Find all sibling elements after this node matching the CSS selector and return a Mojo::Collection object containing these elements as Mojo::DOM objects. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
say $dom ->following-> map ( 'tag' )-> join ( "\n" );
|
following_nodes
my $collection = $dom ->following_nodes;
|
Return a Mojo::Collection object containing all sibling nodes after this node as Mojo::DOM objects.
$dom ->parse( '<p>A</p><!-- B -->C' )->at( 'p' )->following_nodes-> last ->content;
|
matches
my $bool = $dom ->matches( 'div ~ p' );
|
Check if this element matches the CSS selector. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
$dom ->parse( '<p class="a">A</p>' )->at( 'p' )->matches( '.a' );
$dom ->parse( '<p class="a">A</p>' )->at( 'p' )->matches( 'p[class]' );
$dom ->parse( '<p class="a">A</p>' )->at( 'p' )->matches( '.b' );
$dom ->parse( '<p class="a">A</p>' )->at( 'p' )->matches( 'p[id]' );
|
Trailing key/value pairs can be used to declare xml namespace aliases.
namespace
my $namespace = $dom ->namespace;
|
Find this element's namespace, or return undef
if none could be found.
my $namespace = $dom ->at( 'svg > svg\:circle' )->namespace;
my $namespace = $dom ->at( 'svg > circle' )->namespace;
|
new
my $dom = Mojo::DOM->new;
my $dom = Mojo::DOM->new( '<foo bar="baz">I ♥ Mojolicious!</foo>' );
|
Construct a new scalar-based Mojo::DOM object and "parse" HTML/XML fragment if necessary.
new_tag
my $tag = Mojo::DOM->new_tag( 'div' );
my $tag = $dom ->new_tag( 'div' );
my $tag = $dom ->new_tag( 'div' , id => 'foo' , hidden => undef );
my $tag = $dom ->new_tag( 'div' , 'safe content' );
my $tag = $dom ->new_tag( 'div' , id => 'foo' , 'safe content' );
my $tag = $dom ->new_tag( 'div' , data => { mojo => 'rocks' }, 'safe content' );
my $tag = $dom ->new_tag( 'div' , id => 'foo' , sub { 'unsafe content' });
|
Construct a new Mojo::DOM object for an HTML/XML tag with or without attributes and content. The data
attribute may contain a hash reference with key/value pairs to generate attributes from.
$dom ->new_tag( 'br' );
$dom ->new_tag( 'div' );
$dom ->new_tag( 'div' , id => 'foo' , hidden => undef );
$dom ->new_tag( 'div' , 'test & 123' );
$dom ->new_tag( 'div' , id => 'foo' , 'test & 123' );
$dom ->new_tag( 'div' , data => { foo => 1, Bar => 'test' }, 'test & 123' );
$dom ->new_tag( 'div' , id => 'foo' , sub { 'test & 123' });
$dom ->parse( '<div>Hello</div>' )->at( 'div' )
->append_content( $dom ->new_tag( 'b' , 'Mojo!' ))->root;
|
next
my $sibling = $dom -> next ;
|
Return Mojo::DOM object for next sibling element, or undef
if there are no more siblings.
$dom ->parse( '<div><h1>Test</h1><h2>123</h2></div>' )->at( 'h1' )-> next ;
|
next_node
my $sibling = $dom ->next_node;
|
Return Mojo::DOM object for next sibling node, or undef
if there are no more siblings.
$dom ->parse( '<p><b>123</b><!-- Test -->456</p>' )
->at( 'b' )->next_node->next_node;
$dom ->parse( '<p><b>123</b><!-- Test -->456</p>' )
->at( 'b' )->next_node->content;
|
parent
my $parent = $dom ->parent;
|
Return Mojo::DOM object for parent of this node, or undef
if this node has no parent.
$dom ->parse( '<p><b><i>Test</i></b></p>' )->at( 'i' )->parent;
|
parse
$dom = $dom ->parse( '<foo bar="baz">I ♥ Mojolicious!</foo>' );
|
Parse HTML/XML fragment with Mojo::DOM::HTML.
my $dom = Mojo::DOM->new->xml(1)->parse( '<foo>I ♥ Mojolicious!</foo>' );
|
preceding
my $collection = $dom ->preceding;
my $collection = $dom ->preceding( 'div ~ p' );
|
Find all sibling elements before this node matching the CSS selector and return a Mojo::Collection object containing these elements as Mojo::DOM objects. All selectors from "SELECTORS" in Mojo::DOM::CSS are supported.
say $dom ->preceding-> map ( 'tag' )-> join ( "\n" );
|
preceding_nodes
my $collection = $dom ->preceding_nodes;
|
Return a Mojo::Collection object containing all sibling nodes before this node as Mojo::DOM objects.
$dom ->parse( 'A<!-- B --><p>C</p>' )->at( 'p' )->preceding_nodes->first->content;
|
prepend
$dom = $dom ->prepend( '<p>I ♥ Mojolicious!</p>' );
$dom = $dom ->prepend(Mojo::DOM->new);
|
Prepend HTML/XML fragment to this node (for all node types other than root
).
$dom ->parse( '<div><h2>123</h2></div>' )
->at( 'h2' )->prepend( '<h1>Test</h1>' )->root;
$dom ->parse( '<p>123</p>' )
->at( 'p' )->child_nodes->first->prepend( 'Test ' )->root;
|
prepend_content
$dom = $dom ->prepend_content( '<p>I ♥ Mojolicious!</p>' );
$dom = $dom ->prepend_content(Mojo::DOM->new);
|
Prepend HTML/XML fragment (for root
and tag
nodes) or raw content to this node's content.
$dom ->parse( '<div><h2>123</h2></div>' )
->at( 'h2' )->prepend_content( 'Test' )->root;
$dom ->parse( '<!-- 123 --><br>' )
->child_nodes->first->prepend_content( ' Test' )->root;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->prepend_content( '<i>123</i>' )->root;
|
previous
my $sibling = $dom ->previous;
|
Return Mojo::DOM object for previous sibling element, or undef
if there are no more siblings.
$dom ->parse( '<div><h1>Test</h1><h2>123</h2></div>' )->at( 'h2' )->previous;
|
previous_node
my $sibling = $dom ->previous_node;
|
Return Mojo::DOM object for previous sibling node, or undef
if there are no more siblings.
$dom ->parse( '<p>123<!-- Test --><b>456</b></p>' )
->at( 'b' )->previous_node->previous_node;
$dom ->parse( '<p>123<!-- Test --><b>456</b></p>' )
->at( 'b' )->previous_node->content;
|
remove
my $parent = $dom ->remove;
|
Remove this node and return "root" (for root
nodes) or "parent".
$dom ->parse( '<div><h1>Test</h1></div>' )->at( 'h1' )->remove;
$dom ->parse( '<p>123<b>456</b></p>' )
->at( 'p' )->child_nodes->first->remove->root;
|
replace
my $parent = $dom ->replace( '<div>I ♥ Mojolicious!</div>' );
my $parent = $dom ->replace(Mojo::DOM->new);
|
Replace this node with HTML/XML fragment and return "root" (for root
nodes) or "parent".
$dom ->parse( '<div><h1>Test</h1></div>' )->at( 'h1' )->replace( '<h2>123</h2>' );
$dom ->parse( '<p>Test</p>' )
->at( 'p' )->child_nodes->[0]->replace( '<b>123</b>' )->root;
|
root
Return Mojo::DOM object for root
node.
selector
my $selector = $dom ->selector;
|
Get a unique CSS selector for this element.
$dom ->parse( '<ul><li>Test</li><li>123</li></ul>' )->find( 'li' )-> last ->selector;
$dom ->parse( '<p><b><i>Test</i></b></p>' )->at( 'i' )->selector;
|
strip
my $parent = $dom ->strip;
|
Remove this element while preserving its content and return "parent".
$dom ->parse( '<div><h1>Test</h1></div>' )->at( 'h1' )->strip;
|
tag
my $tag = $dom ->tag;
$dom = $dom ->tag( 'div' );
|
This element's tag name.
say $dom ->children-> map ( 'tag' )-> join ( "\n" );
|
tap
$dom = $dom ->tap( sub {...});
|
Alias for "tap" in Mojo::Base.
text
Extract text content from this element only (not including child elements).
$dom ->parse( "<div>foo<p>bar</p>baz</div>" )->at( 'p' )->text;
$dom ->parse( "<div>foo\n<p>bar</p>baz\n</div>" )->at( 'div' )->text;
|
To extract text content from all descendant nodes see "all_text".
to_string
my $str = $dom ->to_string;
|
Render this node and its content to HTML/XML.
$dom ->parse( '<div><b>Test</b></div>' )->at( 'div b' )->to_string;
|
tree
my $tree = $dom ->tree;
$dom = $dom ->tree([ 'root' ]);
|
Document Object Model. Note that this structure should only be used very carefully since it is very dynamic.
type
This node's type, usually cdata
, comment
, doctype
, pi
, raw
, root
, tag
or text
.
$dom ->parse( '<![CDATA[Test]]>' )->child_nodes->first->type;
$dom ->parse( '<!-- Test -->' )->child_nodes->first->type;
$dom ->parse( '<!DOCTYPE html>' )->child_nodes->first->type;
$dom ->parse( '<?xml version="1.0"?>' )->child_nodes->first->type;
$dom ->parse( '<title>Test</title>' )->at( 'title' )->child_nodes->first->type;
$dom ->parse( '<p>Test</p>' )->type;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->type;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->child_nodes->first->type;
|
val
Extract value from form element (such as button
, input
, option
, select
and textarea
), or return undef
if this element has no value. In the case of select
with multiple
attribute, find option
elements with selected
attribute and return an array reference with all values, or undef
if none could be found.
$dom ->parse( '<input name=test value=a>' )->at( 'input' )->val;
$dom ->parse( '<textarea>b</textarea>' )->at( 'textarea' )->val;
$dom ->parse( '<option value="c">Test</option>' )->at( 'option' )->val;
$dom ->parse( '<select><option selected>d</option></select>' )
->at( 'select' )->val;
$dom ->parse( '<select multiple><option selected>e</option></select>' )
->at( 'select' )->val->[0];
$dom ->parse( '<input name=test type=checkbox>' )->at( 'input' )->val;
|
with_roles
my $new_class = Mojo::DOM->with_roles( 'Mojo::DOM::Role::One' );
my $new_class = Mojo::DOM->with_roles( '+One' , '+Two' );
$dom = $dom ->with_roles( '+One' , '+Two' );
|
Alias for "with_roles" in Mojo::Base.
wrap
$dom = $dom ->wrap( '<div></div>' );
$dom = $dom ->wrap(Mojo::DOM->new);
|
Wrap HTML/XML fragment around this node (for all node types other than root
), placing it as the last child of the first innermost element.
$dom ->parse( '<b>Test</b>' )->at( 'b' )->wrap( '<p>123</p>' )->root;
$dom ->parse( '<b>Test</b>' )->at( 'b' )->wrap( '<div><p></p>123</div>' )->root;
$dom ->parse( '<b>Test</b>' )->at( 'b' )->wrap( '<p></p><p>123</p>' )->root;
$dom ->parse( '<p>Test</p>' )->at( 'p' )->child_nodes->first->wrap( '<b>' )->root;
|
wrap_content
$dom = $dom ->wrap_content( '<div></div>' );
$dom = $dom ->wrap_content(Mojo::DOM->new);
|
Wrap HTML/XML fragment around this node's content (for root
and tag
nodes), placing it as the last children of the first innermost element.
$dom ->parse( '<p>Test<p>' )->at( 'p' )->wrap_content( '<b>123</b>' )->root;
$dom ->parse( '<b>Test</b>' )->wrap_content( '<p></p><p>123</p>' );
|
xml
my $bool = $dom ->xml;
$dom = $dom ->xml( $bool );
|
Disable HTML semantics in parser and activate case-sensitivity, defaults to auto-detection based on XML declarations.
OPERATORS
Mojo::DOM overloads the following operators.
array
Alias for "child_nodes".
$dom ->parse( '<!-- Test --><b>123</b>' )->[0];
|
bool
Always true.
hash
Alias for "attr".
$dom ->parse( '<div id="test">Test</div>' )->at( 'div' )->{id};
|
stringify
Alias for "to_string".
SEE ALSO
Mojolicious, Mojolicious::Guides, https://mojolicious.org.