2 * MediaWiki Widgets - CategorySelector class.
4 * @copyright 2011-2015 MediaWiki Widgets Team and others; see AUTHORS.txt
5 * @license The MIT License (MIT); see LICENSE.txt
9 NS_CATEGORY = mw.config.get( 'wgNamespaceIds' ).category;
12 * Category selector widget. Displays an OO.ui.CapsuleMultiSelectWidget
13 * and autocompletes with available categories.
15 * var selector = new mw.widgets.CategorySelector( {
17 * mw.widgets.CategorySelector.SearchType.OpenSearch,
18 * mw.widgets.CategorySelector.SearchType.InternalSearch
22 * $( '#content' ).append( selector.$element );
24 * selector.setSearchType( [ mw.widgets.CategorySelector.SearchType.SubCategories ] );
26 * @class mw.widgets.CategorySelector
28 * @extends OO.ui.CapsuleMultiSelectWidget
29 * @mixins OO.ui.mixin.PendingElement
32 * @param {Object} [config] Configuration options
33 * @cfg {mw.Api} [api] Instance of mw.Api (or subclass thereof) to use for queries
34 * @cfg {number} [limit=10] Maximum number of results to load
35 * @cfg {mw.widgets.CategorySelector.SearchType[]} [searchTypes=[mw.widgets.CategorySelector.SearchType.OpenSearch]]
36 * Default search API to use when searching.
38 function CategorySelector( config ) {
39 // Config initialization
42 searchTypes: [ CategorySelector.SearchType.OpenSearch ]
44 this.limit = config.limit;
45 this.searchTypes = config.searchTypes;
46 this.validateSearchTypes();
49 mw.widgets.CategorySelector.parent.call( this, $.extend( true, {}, config, {
51 filterFromInput: false
53 // This allows the user to both select non-existent categories, and prevents the selector from
54 // being wiped from #onMenuItemsChange when we change the available options in the dropdown
59 OO.ui.mixin.PendingElement.call( this, $.extend( {}, config, { $pending: this.$handle } ) );
61 // Event handler to call the autocomplete methods
62 this.$input.on( 'change input cut paste', OO.ui.debounce( this.updateMenuItems.bind( this ), 100 ) );
65 this.api = config.api || new mw.Api();
70 OO.inheritClass( CategorySelector, OO.ui.CapsuleMultiSelectWidget );
71 OO.mixinClass( CategorySelector, OO.ui.mixin.PendingElement );
72 CSP = CategorySelector.prototype;
77 * Gets new items based on the input by calling
78 * {@link #getNewMenuItems getNewItems} and updates the menu
79 * after removing duplicates based on the data value.
84 CSP.updateMenuItems = function () {
85 this.getMenu().clearItems();
86 this.getNewMenuItems( this.$input.val() ).then( function ( items ) {
87 var existingItems, filteredItems,
88 menu = this.getMenu();
90 // Never show the menu if the input lost focus in the meantime
91 if ( !this.$input.is( ':focus' ) ) {
95 // Array of strings of the data of OO.ui.MenuOptionsWidgets
96 existingItems = menu.getItems().map( function ( item ) {
100 // Remove if items' data already exists
101 filteredItems = items.filter( function ( item ) {
102 return existingItems.indexOf( item ) === -1;
105 // Map to an array of OO.ui.MenuOptionWidgets
106 filteredItems = filteredItems.map( function ( item ) {
107 return new OO.ui.MenuOptionWidget( {
113 menu.addItems( filteredItems ).toggle( true );
120 CSP.clearInput = function () {
121 CategorySelector.parent.prototype.clearInput.call( this );
122 // Abort all pending requests, we won't need their results
127 * Searches for categories based on the input.
131 * @param {string} input The input used to prefix search categories
132 * @return {jQuery.Promise} Resolves with an array of categories
134 CSP.getNewMenuItems = function ( input ) {
137 deferred = $.Deferred();
139 if ( $.trim( input ) === '' ) {
140 deferred.resolve( [] );
141 return deferred.promise();
144 // Abort all pending requests, we won't need their results
146 for ( i = 0; i < this.searchTypes.length; i++ ) {
147 promises.push( this.searchCategories( input, this.searchTypes[ i ] ) );
152 $.when.apply( $, promises ).done( function () {
153 var categories, categoryNames,
155 dataSets = Array.prototype.slice.apply( arguments );
157 // Collect values from all results
158 allData = allData.concat.apply( allData, dataSets );
161 categories = allData.filter( function ( value, index, self ) {
162 return self.indexOf( value ) === index;
166 categoryNames = categories.map( function ( name ) {
167 return mw.Title.newFromText( name, NS_CATEGORY ).getMainText();
170 deferred.resolve( categoryNames );
172 } ).always( this.popPending.bind( this ) );
174 return deferred.promise();
180 CSP.createItemWidget = function ( data ) {
181 return new mw.widgets.CategoryCapsuleItemWidget( {
182 apiUrl: this.api.apiUrl || undefined,
183 title: mw.Title.newFromText( data, NS_CATEGORY )
190 CSP.getItemFromData = function ( data ) {
191 // This is a bit of a hack... We have to canonicalize the data in the same way that
192 // #createItemWidget and CategoryCapsuleItemWidget will do, otherwise we won't find duplicates.
193 data = mw.Title.newFromText( data, NS_CATEGORY ).getMainText();
194 return OO.ui.mixin.GroupElement.prototype.getItemFromData.call( this, data );
198 * Validates the values in `this.searchType`.
203 CSP.validateSearchTypes = function () {
204 var validSearchTypes = false,
205 searchTypeEnumCount = Object.keys( CategorySelector.SearchType ).length;
207 // Check if all values are in the SearchType enum
208 validSearchTypes = this.searchTypes.every( function ( searchType ) {
209 return searchType > -1 && searchType < searchTypeEnumCount;
212 if ( validSearchTypes === false ) {
213 throw new Error( 'Unknown searchType in searchTypes' );
216 // If the searchTypes has CategorySelector.SearchType.SubCategories
217 // it can be the only search type.
218 if ( this.searchTypes.indexOf( CategorySelector.SearchType.SubCategories ) > -1 &&
219 this.searchTypes.length > 1
221 throw new Error( 'Can\'t have additional search types with CategorySelector.SearchType.SubCategories' );
224 // If the searchTypes has CategorySelector.SearchType.ParentCategories
225 // it can be the only search type.
226 if ( this.searchTypes.indexOf( CategorySelector.SearchType.ParentCategories ) > -1 &&
227 this.searchTypes.length > 1
229 throw new Error( 'Can\'t have additional search types with CategorySelector.SearchType.ParentCategories' );
236 * Sets and validates the value of `this.searchType`.
238 * @param {mw.widgets.CategorySelector.SearchType[]} searchTypes
240 CSP.setSearchTypes = function ( searchTypes ) {
241 this.searchTypes = searchTypes;
242 this.validateSearchTypes();
246 * Searches categories based on input and searchType.
250 * @param {string} input The input used to prefix search categories
251 * @param {mw.widgets.CategorySelector.SearchType} searchType
252 * @return {jQuery.Promise} Resolves with an array of categories
254 CSP.searchCategories = function ( input, searchType ) {
255 var deferred = $.Deferred();
257 switch ( searchType ) {
258 case CategorySelector.SearchType.OpenSearch:
260 action: 'opensearch',
261 namespace: NS_CATEGORY,
264 } ).done( function ( res ) {
265 var categories = res[ 1 ];
266 deferred.resolve( categories );
267 } ).fail( deferred.reject.bind( deferred ) );
270 case CategorySelector.SearchType.InternalSearch:
274 apnamespace: NS_CATEGORY,
278 } ).done( function ( res ) {
279 var categories = res.query.allpages.map( function ( page ) {
282 deferred.resolve( categories );
283 } ).fail( deferred.reject.bind( deferred ) );
286 case CategorySelector.SearchType.Exists:
287 if ( input.indexOf( '|' ) > -1 ) {
288 deferred.resolve( [] );
295 titles: 'Category:' + input
296 } ).done( function ( res ) {
300 for ( page in res.query.pages ) {
301 if ( parseInt( page, 10 ) > -1 ) {
302 categories.push( res.query.pages[ page ].title );
306 deferred.resolve( categories );
307 } ).fail( deferred.reject.bind( deferred ) );
310 case CategorySelector.SearchType.SubCategories:
311 if ( input.indexOf( '|' ) > -1 ) {
312 deferred.resolve( [] );
318 list: 'categorymembers',
321 cmtitle: 'Category:' + input
322 } ).done( function ( res ) {
323 var categories = res.query.categorymembers.map( function ( category ) {
324 return category.title;
326 deferred.resolve( categories );
327 } ).fail( deferred.reject.bind( deferred ) );
330 case CategorySelector.SearchType.ParentCategories:
331 if ( input.indexOf( '|' ) > -1 ) {
332 deferred.resolve( [] );
340 titles: 'Category:' + input
341 } ).done( function ( res ) {
345 for ( page in res.query.pages ) {
346 if ( parseInt( page, 10 ) > -1 ) {
347 if ( $.isArray( res.query.pages[ page ].categories ) ) {
348 categories.push.apply( categories, res.query.pages[ page ].categories.map( function ( category ) {
349 return category.title;
355 deferred.resolve( categories );
356 } ).fail( deferred.reject.bind( deferred ) );
360 throw new Error( 'Unknown searchType' );
363 return deferred.promise();
367 * @enum mw.widgets.CategorySelector.SearchType
368 * Types of search available.
370 CategorySelector.SearchType = {
371 /** Search using action=opensearch */
374 /** Search using action=query */
377 /** Search for existing categories with the exact title */
380 /** Search only subcategories */
383 /** Search only parent categories */
387 mw.widgets.CategorySelector = CategorySelector;
388 }( jQuery, mediaWiki ) );