mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-05 22:22:54 +00:00
f9c41c754b
The old regex was lifted from PHP, which matches on UTF-8 byte sequences. In PHP, [...\x80-\xFF]+ matches any bytes with the high bit set, which by the particular properties of UTF-8 will match any sequence which represents a Unicode character above U+007F. In Javascript, regex matching is on UTF-16 Unicode code units, so we don't have to do byte sequence matching (and cannot do so). So the equivalent Javascript regex should use [...\u0080-\uFFFF]+, to match any code unit above U+007F directly. (It also matches surrogate pairs, by the particular properties of UTF-16, so any Unicode codepoint above U+007F is matched). Change-Id: I674b89f757b60331dd1cb23fd7ff8b18775012e9
97 lines
2.5 KiB
JavaScript
97 lines
2.5 KiB
JavaScript
/*!
|
|
* VisualEditor UserInterface LinkInspector class.
|
|
*
|
|
* @copyright 2011-2013 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/*global mw */
|
|
|
|
/**
|
|
* MediaWiki link inspector.
|
|
*
|
|
* @class
|
|
* @extends ve.ui.LinkInspector
|
|
*
|
|
* @constructor
|
|
* @param {ve.ui.Surface} surface
|
|
* @param {Object} [config] Config options
|
|
*/
|
|
ve.ui.MWLinkInspector = function VeUiMWLinkInspector( surface, config ) {
|
|
// Parent constructor
|
|
ve.ui.LinkInspector.call( this, surface, config );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
ve.inheritClass( ve.ui.MWLinkInspector, ve.ui.LinkInspector );
|
|
|
|
/* Static properties */
|
|
|
|
ve.ui.MWLinkInspector.static.modelClasses = [
|
|
ve.dm.MWExternalLinkAnnotation, ve.dm.MWInternalLinkAnnotation
|
|
];
|
|
|
|
ve.ui.MWLinkInspector.static.linkTargetInputWidget = ve.ui.MWLinkTargetInputWidget;
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Gets an annotation object from a target.
|
|
*
|
|
* The type of link is automatically detected based on some crude heuristics.
|
|
*
|
|
* @method
|
|
* @param {string} target Link target
|
|
* @returns {ve.dm.MWInternalLinkAnnotation|ve.dm.MWExternalLinkAnnotation|null}
|
|
*/
|
|
ve.ui.MWLinkInspector.prototype.getAnnotationFromText = function ( target ) {
|
|
var title;
|
|
|
|
// Figure out if this is an internal or external link
|
|
if ( ve.init.platform.getExternalLinkUrlProtocolsRegExp().test( target ) ) {
|
|
// External link
|
|
return new ve.dm.MWExternalLinkAnnotation( {
|
|
'type': 'link/mwExternal',
|
|
'attributes': {
|
|
'href': target
|
|
}
|
|
} );
|
|
} else if ( ve.ui.MWLinkInspector.static.legalTitle.test( target ) ) {
|
|
// Internal link
|
|
// TODO: In the longer term we'll want to have autocompletion and existence and validity
|
|
// checks using AJAX
|
|
try {
|
|
title = new mw.Title( target );
|
|
if ( title.getNamespaceId() === 6 || title.getNamespaceId() === 14 ) {
|
|
// File: or Category: link
|
|
// We have to prepend a colon so this is interpreted as a link
|
|
// rather than an image inclusion or categorization
|
|
target = ':' + target;
|
|
}
|
|
} catch ( e ) { }
|
|
return new ve.dm.MWInternalLinkAnnotation( {
|
|
'type': 'link/mwInternal',
|
|
'attributes': {
|
|
'title': target,
|
|
'normalizedTitle': ve.dm.MWInternalLinkAnnotation.static.normalizeTitle( target )
|
|
}
|
|
} );
|
|
} else {
|
|
return null;
|
|
}
|
|
};
|
|
|
|
/* Static Properties */
|
|
|
|
/**
|
|
* Regular expression matching a valid internal link
|
|
*
|
|
* @type {RegExp}
|
|
*/
|
|
ve.ui.MWLinkInspector.static.legalTitle = /^[ %!"$&'()*,\-.\/0-9:;=?@A-Z\\^_`a-z~\u0080-\uFFFF+]+$/;
|
|
|
|
/* Registration */
|
|
|
|
ve.ui.inspectorFactory.register( 'mwLink', ve.ui.MWLinkInspector );
|