mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/DiscussionTools
synced 2024-11-12 09:58:17 +00:00
fe520ab175
The only use was removed in code review, but we missed this. https://gerrit.wikimedia.org/r/c/mediawiki/extensions/DiscussionTools/+/596813/70..71/includes/ImmutableRange.php Change-Id: Ia761a6a350711c32f0bd4c8af48c7c1a35a4afb4
456 lines
12 KiB
PHP
456 lines
12 KiB
PHP
<?php
|
|
|
|
namespace MediaWiki\Extension\DiscussionTools;
|
|
|
|
use DOMComment;
|
|
use DOMDocumentFragment;
|
|
use DOMDocumentType;
|
|
use DOMNode;
|
|
use DOMProcessingInstruction;
|
|
use DOMText;
|
|
use Error;
|
|
use Exception;
|
|
|
|
/**
|
|
* ImmutableRange has a similar API to the DOM Range class.
|
|
*
|
|
* start/endContainer and offsets can be accessed, as can commonAncestorContainer
|
|
* which is lazy evaluated.
|
|
*
|
|
* setStart and setEnd are still available but return a cloned range.
|
|
*/
|
|
class ImmutableRange {
|
|
private $mCollapsed;
|
|
private $mCommonAncestorContainer;
|
|
private $mEndContainer;
|
|
private $mEndOffset;
|
|
private $mStartContainer;
|
|
private $mStartOffset;
|
|
|
|
/**
|
|
* Find the common ancestor container of two nodes
|
|
*
|
|
* @param DOMNode $a
|
|
* @param DOMNode $b
|
|
* @return DOMNode Common ancestor container
|
|
*/
|
|
private static function findCommonAncestorContainer( DOMNode $a, DOMNode $b ) : DOMNode {
|
|
$ancestorsA = [];
|
|
$ancestorsB = [];
|
|
|
|
do {
|
|
$ancestorsA[] = $a;
|
|
} while ( ( $a = $a->parentNode ) );
|
|
do {
|
|
$ancestorsB[] = $b;
|
|
} while ( ( $b = $b->parentNode ) );
|
|
|
|
$node = null;
|
|
while ( end( $ancestorsA ) && end( $ancestorsA ) === end( $ancestorsB ) ) {
|
|
$node = end( $ancestorsA );
|
|
array_pop( $ancestorsA );
|
|
array_pop( $ancestorsB );
|
|
}
|
|
if ( !$node ) {
|
|
throw new Error( 'Nodes are not in the same document' );
|
|
}
|
|
|
|
return $node;
|
|
}
|
|
|
|
/**
|
|
* Get the root ancestor of a node
|
|
*
|
|
* @param DOMNode $node Node
|
|
* @return DOMNode
|
|
*/
|
|
private static function getRootNode( DOMNode $node ) : DOMNode {
|
|
while ( $node->parentNode ) {
|
|
$node = $node->parentNode;
|
|
}
|
|
|
|
return $node;
|
|
}
|
|
|
|
/**
|
|
* @param DOMNode $startNode Start node
|
|
* @param int $startOffset Start offset
|
|
* @param DOMNode $endNode End node
|
|
* @param int $endOffset End offset
|
|
*/
|
|
public function __construct(
|
|
DOMNode $startNode, int $startOffset, DOMNode $endNode, int $endOffset
|
|
) {
|
|
$this->mStartContainer = $startNode;
|
|
$this->mStartOffset = $startOffset;
|
|
$this->mEndContainer = $endNode;
|
|
$this->mEndOffset = $endOffset;
|
|
}
|
|
|
|
/**
|
|
* @param string $field Field name
|
|
* @return mixed
|
|
*/
|
|
public function __get( string $field ) {
|
|
switch ( $field ) {
|
|
case 'collapsed':
|
|
return $this->mStartContainer === $this->mEndContainer &&
|
|
$this->mStartOffset === $this->mEndOffset;
|
|
case 'commonAncestorContainer':
|
|
if ( !$this->mCommonAncestorContainer ) {
|
|
$this->mCommonAncestorContainer =
|
|
self::findCommonAncestorContainer( $this->mStartContainer, $this->mEndContainer );
|
|
}
|
|
return $this->mCommonAncestorContainer;
|
|
case 'endContainer':
|
|
return $this->mEndContainer;
|
|
case 'endOffset':
|
|
return $this->mEndOffset;
|
|
case 'startContainer':
|
|
return $this->mStartContainer;
|
|
case 'startOffset':
|
|
return $this->mStartOffset;
|
|
default:
|
|
throw new Exception( 'Invalid property: ' . $field );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Clone range with a new start position
|
|
*
|
|
* @param DOMNode $startNode Start node
|
|
* @param int $startOffset Start offset
|
|
* @return self
|
|
*/
|
|
public function setStart( DOMNode $startNode, int $startOffset ) : self {
|
|
return new self(
|
|
$startNode, $startOffset, $this->mEndContainer, $this->mEndOffset
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Clone range with a new end position
|
|
*
|
|
* @param DOMNode $endNode End node
|
|
* @param int $endOffset End offset
|
|
* @return self
|
|
*/
|
|
public function setEnd( DOMNode $endNode, int $endOffset ) : self {
|
|
return new self(
|
|
$this->mStartContainer, $this->mStartOffset, $endNode, $endOffset
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Returns true if only a portion of the Node is contained within the Range.
|
|
*
|
|
* Ported from https://github.com/TRowbotham/PHPDOM (MIT)
|
|
* @see https://dom.spec.whatwg.org/#partially-contained
|
|
*
|
|
* @param DOMNode $node The Node to check against.
|
|
* @return bool
|
|
*/
|
|
private function isPartiallyContainedNode( DOMNode $node ) : bool {
|
|
$isAncestorOfStart = CommentUtils::contains( $node, $this->mStartContainer );
|
|
$isAncestorOfEnd = CommentUtils::contains( $node, $this->mEndContainer );
|
|
|
|
return ( $isAncestorOfStart && !$isAncestorOfEnd )
|
|
|| ( !$isAncestorOfStart && $isAncestorOfEnd );
|
|
}
|
|
|
|
/**
|
|
* Returns true if the entire Node is within the Range, otherwise false.
|
|
*
|
|
* Ported from https://github.com/TRowbotham/PHPDOM (MIT)
|
|
* @see https://dom.spec.whatwg.org/#contained
|
|
*
|
|
* @param DOMNode $node The Node to check against.
|
|
* @return bool
|
|
*/
|
|
private function isFullyContainedNode( DOMNode $node ) : bool {
|
|
$startBP = [ $this->mStartContainer, $this->mStartOffset ];
|
|
$endBP = [ $this->mEndContainer, $this->mEndOffset ];
|
|
$root = self::getRootNode( $this->mStartContainer );
|
|
|
|
return self::getRootNode( $node ) === $root
|
|
&& $this->computePosition( [ $node, 0 ], $startBP ) === 'after'
|
|
&& $this->computePosition(
|
|
[ $node, strlen( $node->nodeValue ) ],
|
|
$endBP
|
|
) === 'before';
|
|
}
|
|
|
|
/**
|
|
* Ported from https://github.com/TRowbotham/PHPDOM (MIT)
|
|
* @see https://dom.spec.whatwg.org/#dom-range-clonecontents
|
|
*
|
|
* @return DOMDocumentFragment
|
|
*/
|
|
public function cloneContents() : DOMDocumentFragment {
|
|
$ownerDocument = $this->mStartContainer->ownerDocument;
|
|
$fragment = $ownerDocument->createDocumentFragment();
|
|
|
|
if ( $this->mStartContainer === $this->mEndContainer
|
|
&& $this->mStartOffset === $this->mEndOffset
|
|
) {
|
|
return $fragment;
|
|
}
|
|
|
|
$originalStartContainer = $this->mStartContainer;
|
|
$originalStartOffset = $this->mStartOffset;
|
|
$originalEndContainer = $this->mEndContainer;
|
|
$originalEndOffset = $this->mEndOffset;
|
|
|
|
if ( $originalStartContainer === $originalEndContainer
|
|
&& ( $originalStartContainer instanceof DOMText
|
|
|| $originalStartContainer instanceof DOMProcessingInstruction
|
|
|| $originalStartContainer instanceof DOMComment )
|
|
) {
|
|
$clone = $originalStartContainer->cloneNode();
|
|
$clone->nodeValue = substr(
|
|
$originalStartContainer->nodeValue,
|
|
$originalStartOffset,
|
|
$originalEndOffset - $originalStartOffset
|
|
);
|
|
$fragment->appendChild( $clone );
|
|
|
|
return $fragment;
|
|
}
|
|
|
|
$commonAncestor = self::findCommonAncestorContainer(
|
|
$originalStartContainer,
|
|
$originalEndContainer
|
|
);
|
|
$firstPartiallyContainedChild = null;
|
|
|
|
if ( !CommentUtils::contains( $originalStartContainer, $originalEndContainer ) ) {
|
|
foreach ( $commonAncestor->childNodes as $node ) {
|
|
if ( $this->isPartiallyContainedNode( $node ) ) {
|
|
$firstPartiallyContainedChild = $node;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
$lastPartiallyContainedChild = null;
|
|
|
|
if ( !CommentUtils::contains( $originalEndContainer, $originalStartContainer ) ) {
|
|
$childNodes = iterator_to_array( $commonAncestor->childNodes );
|
|
|
|
foreach ( array_reverse( $childNodes ) as $node ) {
|
|
if ( $this->isPartiallyContainedNode( $node ) ) {
|
|
$lastPartiallyContainedChild = $node;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
$containedChildren = [];
|
|
|
|
foreach ( $commonAncestor->childNodes as $child ) {
|
|
if ( $this->isFullyContainedNode( $child ) ) {
|
|
if ( $child instanceof DOMDocumentType ) {
|
|
throw new Error();
|
|
}
|
|
|
|
$containedChildren[] = $child;
|
|
}
|
|
}
|
|
|
|
if ( $firstPartiallyContainedChild instanceof DOMText
|
|
|| $firstPartiallyContainedChild instanceof DOMProcessingInstruction
|
|
|| $firstPartiallyContainedChild instanceof DOMComment
|
|
) {
|
|
$clone = $originalStartContainer->cloneNode();
|
|
$clone->nodeValue = substr(
|
|
$originalStartContainer->nodeValue,
|
|
$originalStartOffset,
|
|
strlen( $originalStartContainer->nodeValue ) - $originalStartOffset
|
|
);
|
|
$fragment->appendChild( $clone );
|
|
} elseif ( $firstPartiallyContainedChild ) {
|
|
$clone = $firstPartiallyContainedChild->cloneNode();
|
|
$fragment->appendChild( $clone );
|
|
$subrange = new self(
|
|
$originalStartContainer, $originalStartOffset,
|
|
$firstPartiallyContainedChild,
|
|
strlen( $firstPartiallyContainedChild->nodeValue )
|
|
);
|
|
$subfragment = $subrange->cloneContents();
|
|
$clone->appendChild( $subfragment );
|
|
}
|
|
|
|
foreach ( $containedChildren as $child ) {
|
|
$clone = $child->cloneNode( true );
|
|
$fragment->appendChild( $clone );
|
|
}
|
|
|
|
if ( $lastPartiallyContainedChild instanceof DOMText
|
|
|| $lastPartiallyContainedChild instanceof DOMProcessingInstruction
|
|
|| $lastPartiallyContainedChild instanceof DOMComment
|
|
) {
|
|
$clone = $originalEndContainer->cloneNode();
|
|
$clone->nodeValue = substr(
|
|
$originalEndContainer->nodeValue,
|
|
0,
|
|
$originalEndOffset
|
|
);
|
|
$fragment->appendChild( $clone );
|
|
} elseif ( $lastPartiallyContainedChild ) {
|
|
$clone = $lastPartiallyContainedChild->cloneNode();
|
|
$fragment->appendChild( $clone );
|
|
$subrange = new self(
|
|
$lastPartiallyContainedChild, 0,
|
|
$originalEndContainer, $originalEndOffset
|
|
);
|
|
$subfragment = $subrange->cloneContents();
|
|
$clone->appendChild( $subfragment );
|
|
}
|
|
|
|
return $fragment;
|
|
}
|
|
|
|
/**
|
|
* Inserts a new Node into at the start of the Range.
|
|
*
|
|
* Ported from https://github.com/TRowbotham/PHPDOM (MIT)
|
|
*
|
|
* @see https://dom.spec.whatwg.org/#dom-range-insertnode
|
|
*
|
|
* @param DOMNode $node The Node to be inserted.
|
|
* @return void
|
|
*/
|
|
public function insertNode( DOMNode $node ) : void {
|
|
if ( ( $this->mStartContainer instanceof DOMProcessingInstruction
|
|
|| $this->mStartContainer instanceof DOMComment )
|
|
|| ( $this->mStartContainer instanceof DOMText
|
|
&& $this->mStartContainer->parentNode === null )
|
|
) {
|
|
throw new Error();
|
|
}
|
|
|
|
$referenceNode = null;
|
|
|
|
if ( $this->mStartContainer instanceof DOMText ) {
|
|
$referenceNode = $this->mStartContainer;
|
|
} else {
|
|
$referenceNode = $this
|
|
->mStartContainer
|
|
->childNodes
|
|
->item( $this->mStartOffset );
|
|
}
|
|
|
|
$parent = !$referenceNode
|
|
? $this->mStartContainer
|
|
: $referenceNode->parentNode;
|
|
// TODO: Restore this validation check?
|
|
// $parent->ensurePreinsertionValidity( $node, $referenceNode );
|
|
|
|
if ( $this->mStartContainer instanceof DOMText ) {
|
|
$referenceNode = $this->mStartContainer->splitText( $this->mStartOffset );
|
|
}
|
|
|
|
if ( $node === $referenceNode ) {
|
|
$referenceNode = $referenceNode->nextSibling;
|
|
}
|
|
|
|
if ( $node->parentNode ) {
|
|
$node->parentNode->removeChild( $node );
|
|
}
|
|
|
|
// TODO: Restore this validation check?
|
|
// $parent->preinsertNode( $node, $referenceNode );
|
|
//
|
|
// This should just be
|
|
// $parent->insertBefore( $node, $referenceNode );
|
|
// but the second argument is optional, not nullable
|
|
if ( $referenceNode ) {
|
|
$parent->insertBefore( $node, $referenceNode );
|
|
} else {
|
|
$parent->insertBefore( $node );
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Compares the position of two boundary points.
|
|
*
|
|
* Ported from https://github.com/TRowbotham/PHPDOM (MIT)
|
|
* @internal
|
|
*
|
|
* @see https://dom.spec.whatwg.org/#concept-range-bp-position
|
|
*
|
|
* @param mixed[] $boundaryPointA An array containing a Node and an offset within that Node representing a boundary.
|
|
* @param mixed[] $boundaryPointB An array containing a Node and an offset within that Node representing a boundary.
|
|
* @return string Returns before, equal, or after based on the position of the first boundary relative to the second
|
|
* boundary.
|
|
*/
|
|
private function computePosition(
|
|
array $boundaryPointA,
|
|
array $boundaryPointB
|
|
) : string {
|
|
if ( $boundaryPointA[0] === $boundaryPointB[0] ) {
|
|
if ( $boundaryPointA[1] === $boundaryPointB[1] ) {
|
|
return 'equal';
|
|
} elseif ( $boundaryPointA[1] < $boundaryPointB[1] ) {
|
|
return 'before';
|
|
} else {
|
|
return 'after';
|
|
}
|
|
}
|
|
|
|
$tw = new TreeWalker(
|
|
self::getRootNode( $boundaryPointB[0] ),
|
|
NodeFilter::SHOW_ALL,
|
|
function ( $node ) use ( $boundaryPointA ) {
|
|
if ( $node === $boundaryPointA[0] ) {
|
|
return NodeFilter::FILTER_ACCEPT;
|
|
}
|
|
|
|
return NodeFilter::FILTER_SKIP;
|
|
}
|
|
);
|
|
$tw->currentNode = $boundaryPointB[0];
|
|
|
|
$AFollowsB = $tw->nextNode();
|
|
|
|
if ( $AFollowsB ) {
|
|
switch ( $this->computePosition( $boundaryPointB, $boundaryPointA ) ) {
|
|
case 'after':
|
|
return 'before';
|
|
case 'before':
|
|
return 'after';
|
|
}
|
|
}
|
|
|
|
$ancestor = $boundaryPointB[0]->parentNode;
|
|
|
|
while ( $ancestor ) {
|
|
if ( $ancestor === $boundaryPointA[0] ) {
|
|
break;
|
|
}
|
|
|
|
$ancestor = $ancestor->parentNode;
|
|
}
|
|
|
|
if ( $ancestor ) {
|
|
$child = $boundaryPointB[0];
|
|
$childNodes = iterator_to_array( $boundaryPointA[0]->childNodes );
|
|
|
|
while ( $child ) {
|
|
if ( in_array( $child, $childNodes, true ) ) {
|
|
break;
|
|
}
|
|
|
|
$child = $child->parentNode;
|
|
}
|
|
|
|
if ( CommentUtils::childIndexOf( $child ) < $boundaryPointA[1] ) {
|
|
return 'after';
|
|
}
|
|
}
|
|
|
|
return 'before';
|
|
}
|
|
|
|
}
|