2020-05-22 16:26:05 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace MediaWiki\Extension\DiscussionTools;
|
|
|
|
|
|
|
|
class HeadingItem extends ThreadItem {
|
|
|
|
private $placeholderHeading = false;
|
2020-10-01 19:36:11 +00:00
|
|
|
private $headingLevel;
|
2020-05-22 16:26:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param ImmutableRange $range
|
2020-10-01 19:36:11 +00:00
|
|
|
* @param int $headingLevel Heading level (1-6)
|
2020-06-25 12:23:17 +00:00
|
|
|
* @param bool $placeholderHeading Item doesn't correspond to a real heading (e.g. 0th section)
|
2020-05-22 16:26:05 +00:00
|
|
|
*/
|
|
|
|
public function __construct(
|
2020-10-01 19:36:11 +00:00
|
|
|
ImmutableRange $range, int $headingLevel, bool $placeholderHeading = false
|
2020-05-22 16:26:05 +00:00
|
|
|
) {
|
|
|
|
parent::__construct( 'heading', 0, $range );
|
2020-10-01 19:36:11 +00:00
|
|
|
$this->headingLevel = $headingLevel;
|
2020-05-22 16:26:05 +00:00
|
|
|
$this->placeholderHeading = $placeholderHeading;
|
|
|
|
}
|
|
|
|
|
2020-09-16 12:06:14 +00:00
|
|
|
/**
|
|
|
|
* @return array JSON-serializable array
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function jsonSerialize(): array {
|
2020-09-16 12:06:14 +00:00
|
|
|
return array_merge( parent::jsonSerialize(), [
|
2020-10-01 19:36:11 +00:00
|
|
|
'headingLevel' => $this->headingLevel,
|
2020-09-16 12:06:14 +00:00
|
|
|
'placeholderHeading' => $this->placeholderHeading,
|
|
|
|
] );
|
|
|
|
}
|
|
|
|
|
2020-11-05 16:07:56 +00:00
|
|
|
/**
|
|
|
|
* Get a title based on the hash ID, such that it can be linked to
|
|
|
|
*
|
|
|
|
* @return string Title
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function getLinkableTitle(): string {
|
2020-11-05 16:07:56 +00:00
|
|
|
$title = '';
|
|
|
|
// If this comment is in 0th section, there's no section title for the edit summary
|
|
|
|
if ( !$this->isPlaceholderHeading() ) {
|
|
|
|
$headingNode =
|
|
|
|
CommentUtils::getHeadlineNodeAndOffset( $this->getRange()->startContainer )['node'];
|
|
|
|
$id = $headingNode->getAttribute( 'id' );
|
|
|
|
if ( $id ) {
|
|
|
|
// Replace underscores with spaces to undo Sanitizer::escapeIdInternal().
|
|
|
|
// This assumes that $wgFragmentMode is [ 'html5', 'legacy' ] or [ 'html5' ],
|
|
|
|
// otherwise the escaped IDs are super garbled and can't be unescaped reliably.
|
|
|
|
$title = str_replace( '_', ' ', $id );
|
|
|
|
}
|
|
|
|
// else: Not a real section, probably just HTML markup in wikitext
|
|
|
|
}
|
|
|
|
return $title;
|
|
|
|
}
|
|
|
|
|
2020-10-01 19:36:11 +00:00
|
|
|
/**
|
|
|
|
* @return int Heading level (1-6)
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function getHeadingLevel(): int {
|
2020-10-01 19:36:11 +00:00
|
|
|
return $this->headingLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param int $headingLevel Heading level (1-6)
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function setHeadingLevel( int $headingLevel ): void {
|
2020-10-01 19:36:11 +00:00
|
|
|
$this->headingLevel = $headingLevel;
|
|
|
|
}
|
|
|
|
|
2020-05-22 16:26:05 +00:00
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function isPlaceholderHeading(): bool {
|
2020-05-22 16:26:05 +00:00
|
|
|
return $this->placeholderHeading;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param bool $placeholderHeading
|
|
|
|
*/
|
2021-07-22 07:25:13 +00:00
|
|
|
public function setPlaceholderHeading( bool $placeholderHeading ): void {
|
2020-05-22 16:26:05 +00:00
|
|
|
$this->placeholderHeading = $placeholderHeading;
|
|
|
|
}
|
2021-09-01 22:05:00 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check whether this heading can be used for topic subscriptions.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function isSubscribable(): bool {
|
|
|
|
return (
|
|
|
|
// Placeholder headings have nothing to attach the button to.
|
|
|
|
!$this->isPlaceholderHeading() &&
|
|
|
|
// We only allow subscribing to level 2 headings, because the user interface for sub-headings
|
|
|
|
// would be difficult to present.
|
|
|
|
$this->getHeadingLevel() === 2 &&
|
|
|
|
// Check if the name corresponds to a section that contain no comments (only sub-sections).
|
|
|
|
// They can't be distinguished from each other, so disallow subscribing.
|
|
|
|
$this->getName() !== 'h-'
|
|
|
|
);
|
|
|
|
}
|
2022-01-11 21:58:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
public function getTranscludedFrom() {
|
|
|
|
// Placeholder headings break the usual logic, because their ranges are collapsed
|
|
|
|
if ( $this->isPlaceholderHeading() ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return parent::getTranscludedFrom();
|
|
|
|
}
|
2020-05-22 16:26:05 +00:00
|
|
|
}
|