2016-06-14 22:27:44 +00:00
|
|
|
/*!
|
|
|
|
* VisualEditor MediaWiki utilities.
|
|
|
|
*
|
2019-01-01 13:24:23 +00:00
|
|
|
* @copyright 2011-2019 VisualEditor Team and others; see http://ve.mit-license.org
|
2016-06-14 22:27:44 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @class ve
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decode a URI component into a mediawiki article title
|
|
|
|
*
|
|
|
|
* N.B. Illegal article titles can result from fairly reasonable input (e.g. "100%25beef");
|
|
|
|
* see https://phabricator.wikimedia.org/T137847 .
|
|
|
|
*
|
|
|
|
* @param {string} s String to decode
|
|
|
|
* @param {boolean} [preserveUnderscores] Don't convert underscores to spaces
|
|
|
|
* @return {string} Decoded string, or original string if decodeURIComponent failed
|
|
|
|
*/
|
|
|
|
ve.decodeURIComponentIntoArticleTitle = function ( s, preserveUnderscores ) {
|
|
|
|
try {
|
|
|
|
s = decodeURIComponent( s );
|
|
|
|
} catch ( e ) {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
if ( preserveUnderscores ) {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
return s.replace( /_/g, ' ' );
|
|
|
|
};
|
2017-10-26 19:27:21 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Unwrap Parsoid sections
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Parent element, e.g. document body
|
2019-02-13 13:21:26 +00:00
|
|
|
* @param {number} [keepSection] Section to keep
|
2017-10-26 19:27:21 +00:00
|
|
|
*/
|
2019-02-13 13:21:26 +00:00
|
|
|
ve.unwrapParsoidSections = function ( element, keepSection ) {
|
2017-10-26 19:27:21 +00:00
|
|
|
Array.prototype.forEach.call( element.querySelectorAll( 'section[data-mw-section-id]' ), function ( section ) {
|
2019-02-13 13:21:26 +00:00
|
|
|
var parent = section.parentNode,
|
|
|
|
sectionId = section.getAttribute( 'data-mw-section-id' );
|
|
|
|
// Copy section ID to first child (should be a heading)
|
|
|
|
if ( sectionId > 0 ) {
|
|
|
|
section.firstChild.setAttribute( 'data-mw-section-id', sectionId );
|
|
|
|
}
|
|
|
|
if ( keepSection !== undefined && +sectionId === keepSection ) {
|
|
|
|
return;
|
|
|
|
}
|
2017-10-26 19:27:21 +00:00
|
|
|
while ( section.firstChild ) {
|
|
|
|
parent.insertBefore( section.firstChild, section );
|
|
|
|
}
|
|
|
|
parent.removeChild( section );
|
|
|
|
} );
|
|
|
|
};
|
2017-12-06 20:24:49 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Strip legacy (non-HTML5) IDs; typically found as section IDs inside
|
|
|
|
* headings.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Parent element, e.g. document body
|
|
|
|
*/
|
|
|
|
ve.stripParsoidFallbackIds = function ( element ) {
|
|
|
|
Array.prototype.forEach.call( element.querySelectorAll( 'span[typeof="mw:FallbackId"][id]:empty' ), function ( legacySpan ) {
|
|
|
|
legacySpan.parentNode.removeChild( legacySpan );
|
|
|
|
} );
|
|
|
|
};
|
2017-08-13 19:22:04 +00:00
|
|
|
|
2019-01-03 17:11:29 +00:00
|
|
|
/**
|
|
|
|
* Fix fragment links which should be relative to the current document
|
|
|
|
*
|
|
|
|
* This prevents these links from trying to navigate to another page,
|
|
|
|
* or open in a new window.
|
|
|
|
*
|
|
|
|
* Call this after ve.targetLinksToNewWindow, as it removes the target attribute.
|
|
|
|
* Call this after LinkCache.styleParsoidElements, as it breaks that method by including the query string.
|
|
|
|
*
|
|
|
|
* @param {HTMLElement} element Parent element, e.g. document body
|
|
|
|
* @param {mw.Title} title Current title, only links to this title will be normalized
|
|
|
|
* @param {string} [prefix] Prefix to add to fragment and target ID to avoid collisions
|
|
|
|
*/
|
|
|
|
ve.fixFragmentLinks = function ( container, docTitle, prefix ) {
|
|
|
|
var docTitleText = docTitle.getPrefixedText();
|
|
|
|
prefix = prefix || '';
|
|
|
|
Array.prototype.forEach.call( container.querySelectorAll( 'a[href*="#"]' ), function ( el ) {
|
|
|
|
var target, title,
|
|
|
|
fragment = new mw.Uri( el.href ).fragment,
|
|
|
|
targetData = ve.dm.MWInternalLinkAnnotation.static.getTargetDataFromHref( el.href, el.ownerDocument );
|
|
|
|
|
|
|
|
if ( targetData.isInternal ) {
|
|
|
|
title = mw.Title.newFromText( targetData.title );
|
|
|
|
if ( title && title.getPrefixedText() === docTitleText ) {
|
|
|
|
|
|
|
|
if ( !fragment ) {
|
|
|
|
// Special case for empty fragment, even if prefix set
|
|
|
|
el.setAttribute( 'href', '#' );
|
|
|
|
} else {
|
|
|
|
if ( prefix ) {
|
|
|
|
target = container.querySelector( '#' + $.escapeSelector( fragment ) );
|
|
|
|
// There may be multiple links to a specific target, so check the target
|
|
|
|
// hasn't already been fixed (in which case it would be null)
|
|
|
|
if ( target ) {
|
|
|
|
target.setAttribute( 'id', prefix + fragment );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
el.setAttribute( 'href', '#' + prefix + fragment );
|
|
|
|
}
|
|
|
|
el.removeAttribute( 'target' );
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} );
|
|
|
|
};
|
|
|
|
|
2017-08-13 19:22:04 +00:00
|
|
|
/**
|
|
|
|
* Expand a string of the form jquery.foo,bar|jquery.ui.baz,quux to
|
|
|
|
* an array of module names like [ 'jquery.foo', 'jquery.bar',
|
|
|
|
* 'jquery.ui.baz', 'jquery.ui.quux' ]
|
|
|
|
*
|
|
|
|
* Implementation of ResourceLoaderContext::expandModuleNames
|
|
|
|
* TODO: Consider upstreaming this to MW core.
|
|
|
|
*
|
|
|
|
* @param {string} moduleNames Packed module name list
|
|
|
|
* @return {string[]} Array of module names
|
|
|
|
*/
|
|
|
|
ve.expandModuleNames = function ( moduleNames ) {
|
|
|
|
var modules = [];
|
|
|
|
|
|
|
|
moduleNames.split( '|' ).forEach( function ( group ) {
|
|
|
|
var matches, prefix, suffixes;
|
|
|
|
if ( group.indexOf( ',' ) === -1 ) {
|
|
|
|
// This is not a set of modules in foo.bar,baz notation
|
|
|
|
// but a single module
|
|
|
|
modules.push( group );
|
|
|
|
} else {
|
|
|
|
// This is a set of modules in foo.bar,baz notation
|
|
|
|
matches = group.match( /(.*)\.([^.]*)/ );
|
|
|
|
if ( !matches ) {
|
|
|
|
// Prefixless modules, i.e. without dots
|
|
|
|
modules = modules.concat( group.split( ',' ) );
|
|
|
|
} else {
|
|
|
|
// We have a prefix and a bunch of suffixes
|
|
|
|
prefix = matches[ 1 ];
|
|
|
|
suffixes = matches[ 2 ].split( ',' ); // [ 'bar', 'baz' ]
|
|
|
|
suffixes.forEach( function ( suffix ) {
|
|
|
|
modules.push( prefix + '.' + suffix );
|
|
|
|
} );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} );
|
|
|
|
return modules;
|
|
|
|
};
|
2018-02-28 02:01:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Split Parsoid resource name into the href prefix and the page title.
|
|
|
|
*
|
|
|
|
* @param {string} resourceName Resource name, from a `href` or `resource` attribute
|
|
|
|
* @return {Object} Object with the following properties:
|
|
|
|
* @return {string} return.title Full page title in text form (with namespace, and spaces instead of underscores)
|
|
|
|
* @return {string} return.hrefPrefix Href prefix like './' or '../'
|
|
|
|
* @return {string} return.rawTitle Everything following `hrefPrefix` in input, unprocessed
|
|
|
|
*/
|
|
|
|
ve.parseParsoidResourceName = function ( resourceName ) {
|
|
|
|
// Resource names are always prefixed with './' to prevent the MediaWiki namespace from being
|
|
|
|
// interpreted as a URL protocol, consider e.g. 'href="./File:Foo.png"'. If this resource name
|
|
|
|
// came from a page that is a subpage, it is also prefixed with appropriate number of '../'.
|
|
|
|
// (We accept input without the prefix, so this can also take plain page titles.)
|
|
|
|
var matches = resourceName.match( /^((?:\.\.?\/)*)(.*)$/ );
|
|
|
|
return {
|
|
|
|
// '%' and '?' are valid in page titles, but normally URI-encoded. This also changes underscores
|
|
|
|
// to spaces.
|
|
|
|
title: ve.decodeURIComponentIntoArticleTitle( matches[ 2 ] ),
|
|
|
|
rawTitle: matches[ 2 ],
|
|
|
|
hrefPrefix: matches[ 1 ]
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extract the page title from a Parsoid resource name.
|
|
|
|
*
|
|
|
|
* @param {string} resourceName Resource name, from a `href` or `resource` attribute
|
|
|
|
* @return {string} Full page title in text form (with namespace, and spaces instead of underscores)
|
|
|
|
*/
|
|
|
|
ve.normalizeParsoidResourceName = function ( resourceName ) {
|
|
|
|
return ve.parseParsoidResourceName( resourceName ).title;
|
|
|
|
};
|