mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/VisualEditor
synced 2024-11-30 00:55:00 +00:00
0e51180709
Move various bits of code from initialize to setup as required. MWDialog is now useless so use ve.ui.Dialog directly. New changes: 0b2d03f [BREAKING CHANGE] Pass fragments, not surfaces, to windows (reprise) 0101ac4 Update OOjs UI to v0.1.0-pre (ac6848398c) Change-Id: I0fae3e5ff2bd0da5b74c244af226b07b1f5883bd
187 lines
4.7 KiB
JavaScript
187 lines
4.7 KiB
JavaScript
/*!
|
|
* VisualEditor UserInterface MWReferenceSearchWidget class.
|
|
*
|
|
* @copyright 2011-2014 VisualEditor Team and others; see AUTHORS.txt
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
*/
|
|
|
|
/**
|
|
* Creates an ve.ui.MWReferenceSearchWidget object.
|
|
*
|
|
* @class
|
|
* @extends OO.ui.SearchWidget
|
|
*
|
|
* @constructor
|
|
* @param {Object} [config] Configuration options
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget = function VeUiMWReferenceSearchWidget( config ) {
|
|
// Configuration intialization
|
|
config = ve.extendObject( {
|
|
'placeholder': ve.msg( 'visualeditor-reference-input-placeholder' )
|
|
}, config );
|
|
|
|
// Parent constructor
|
|
OO.ui.SearchWidget.call( this, config );
|
|
|
|
// Properties
|
|
this.index = [];
|
|
|
|
// Initialization
|
|
this.$element.addClass( 've-ui-mwReferenceSearchWidget' );
|
|
};
|
|
|
|
/* Inheritance */
|
|
|
|
OO.inheritClass( ve.ui.MWReferenceSearchWidget, OO.ui.SearchWidget );
|
|
|
|
/* Events */
|
|
|
|
/**
|
|
* @event select
|
|
* @param {ve.dm.MWReferenceModel|null} data Reference model, null if no item is selected
|
|
*/
|
|
|
|
/* Methods */
|
|
|
|
/**
|
|
* Handle query change events.
|
|
*
|
|
* @method
|
|
* @param {string} value New value
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget.prototype.onQueryChange = function () {
|
|
// Parent method
|
|
OO.ui.SearchWidget.prototype.onQueryChange.call( this );
|
|
|
|
// Populate
|
|
this.addResults();
|
|
};
|
|
|
|
/**
|
|
* Handle select widget select events.
|
|
*
|
|
* @method
|
|
* @param {OO.ui.OptionWidget} item Selected item
|
|
* @fires select
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget.prototype.onResultsSelect = function ( item ) {
|
|
var data;
|
|
|
|
if ( item ) {
|
|
data = item.getData();
|
|
// Resolve indexed values
|
|
if ( this.index[data] ) {
|
|
data = this.index[data].reference;
|
|
}
|
|
} else {
|
|
data = null;
|
|
}
|
|
|
|
this.emit( 'select', data );
|
|
};
|
|
|
|
/**
|
|
* Build a serchable index of references.
|
|
*
|
|
* @method
|
|
* @param {ve.dm.InternalList} internalList Internal list
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget.prototype.buildIndex = function ( internalList ) {
|
|
var i, iLen, j, jLen, ref, group, groupName, groupNames, view, text, firstNodes, indexOrder,
|
|
refGroup, refNode, matches, name, citation,
|
|
groups = internalList.getNodeGroups();
|
|
|
|
function extractAttrs() {
|
|
text += ' ' + this.getAttribute( 'href' );
|
|
}
|
|
|
|
this.index = [];
|
|
groupNames = ve.getObjectKeys( groups ).sort();
|
|
|
|
for ( i = 0, iLen = groupNames.length; i < iLen; i++ ) {
|
|
groupName = groupNames[i];
|
|
if ( groupName.lastIndexOf( 'mwReference/' ) !== 0 ) {
|
|
continue;
|
|
}
|
|
group = groups[groupName];
|
|
firstNodes = group.firstNodes;
|
|
indexOrder = group.indexOrder;
|
|
for ( j = 0, jLen = indexOrder.length; j < jLen; j++ ) {
|
|
refNode = firstNodes[indexOrder[j]];
|
|
ref = ve.dm.MWReferenceModel.static.newFromReferenceNode( refNode );
|
|
view = new ve.ce.InternalItemNode( internalList.getItemNode( ref.getListIndex() ) );
|
|
|
|
// HACK: PHP parser doesn't wrap single lines in a paragraph
|
|
if ( view.$element.children().length === 1 && view.$element.children( 'p' ).length === 1 ) {
|
|
// unwrap inner
|
|
view.$element.children().replaceWith( view.$element.children().contents() );
|
|
}
|
|
|
|
refGroup = ref.getGroup();
|
|
citation = ( refGroup && refGroup.length ? refGroup + ' ' : '' ) + ( j + 1 );
|
|
matches = ref.getListKey().match( /^literal\/(.*)$/ );
|
|
name = matches && matches[1] || '';
|
|
// Hide previously auto-generated reference names
|
|
if ( name.match( /^:[0-9]+$/ ) ) {
|
|
name = '';
|
|
}
|
|
// Make visible text, citation and reference name searchable
|
|
text = [ view.$element.text().toLowerCase(), citation, name ].join( ' ' );
|
|
// Make URLs searchable
|
|
view.$element.find( 'a[href]' ).each( extractAttrs );
|
|
|
|
this.index.push( {
|
|
'$element': view.$element.clone().show(),
|
|
'text': text,
|
|
'reference': ref,
|
|
'citation': citation,
|
|
'name': name
|
|
} );
|
|
view.destroy();
|
|
}
|
|
}
|
|
|
|
// Re-populate
|
|
this.onQueryChange();
|
|
};
|
|
|
|
/**
|
|
* Check whether the index built by #buildIndex is empty. This will return true if
|
|
* #buildIndex hasn't been called yet.
|
|
* @returns {boolean} Index is empty
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget.prototype.isIndexEmpty = function () {
|
|
return this.index.length === 0;
|
|
};
|
|
|
|
/**
|
|
* Handle media query response events.
|
|
*
|
|
* @method
|
|
*/
|
|
ve.ui.MWReferenceSearchWidget.prototype.addResults = function () {
|
|
var i, len, item, $citation, $name,
|
|
value = this.query.getValue(),
|
|
query = value.toLowerCase(),
|
|
items = [];
|
|
|
|
for ( i = 0, len = this.index.length; i < len; i++ ) {
|
|
item = this.index[i];
|
|
if ( item.text.indexOf( query ) >= 0 ) {
|
|
$citation = this.$( '<div>' )
|
|
.addClass( 've-ui-mwReferenceSearchWidget-citation' )
|
|
.text( '[' + item.citation + ']' );
|
|
$name = this.$( '<div>' )
|
|
.addClass( 've-ui-mwReferenceSearchWidget-name' )
|
|
.text( item.name );
|
|
items.push(
|
|
new ve.ui.MWReferenceResultWidget( i, {
|
|
'$': this.$, 'label': $citation.add( $name ).add( item.$element )
|
|
} )
|
|
);
|
|
}
|
|
}
|
|
|
|
this.results.addItems( items );
|
|
};
|