use warnings;
use strict;
use XML::LibXML;
my $dom = XML::LibXML->load_xml(string => <<'END_XML');
FooB
a
r
END_XML
my $xpc = XML::LibXML::XPathContext->new($dom);
$xpc->registerNs('office',
'urn:oasis:names:tc:opendocument:xmlns:office:1.0');
$xpc->registerNs('text',
'urn:oasis:names:tc:opendocument:xmlns:text:1.0');
my $breakat = 'text:line-break';
my $ancestor = 'text:p';
while (1) {
my ($br1) = $xpc->findnodes("//$ancestor//${breakat}[1]") or last;
die "can't handle <$breakat> with children: $br1"
if $br1->hasChildNodes;
my ($an1) = $xpc->findnodes("ancestor::${ancestor}[1]",$br1)
or die "failed to find <$ancestor> ancestor of $br1";
my $an2 = $an1->cloneNode(1);
my ($br2) = $xpc->findnodes(".//${breakat}[1]", $an2)
or die "internal error: failed to find <$breakat> in $an2";
for ( my $cur = $br1; $cur!=$an1 ; $cur = $cur->parentNode ) {
while ( my $s = $cur->nextSibling )
{ $s->unbindNode }
}
$br1->unbindNode;
for ( my $cur = $br2; $cur!=$an2 ; $cur = $cur->parentNode ) {
while ( my $s = $cur->previousSibling )
{ $s->unbindNode }
}
$br2->unbindNode;
$an1->parentNode->insertAfter($an2, $an1);
}
print $dom;
__END__
FooB
a
r