2021-07-13 09:00:57 +00:00
|
|
|
/**
|
2021-07-13 19:01:30 +00:00
|
|
|
* Generic button-like widget for items in the template dialog sidebar. See
|
|
|
|
* {@see OO.ui.ButtonWidget} for inspiration.
|
2021-07-13 10:15:47 +00:00
|
|
|
*
|
2021-07-13 09:00:57 +00:00
|
|
|
* @class
|
2021-08-27 16:22:37 +00:00
|
|
|
* @extends OO.ui.OptionWidget
|
2021-07-13 09:00:57 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* @param {Object} config
|
2021-09-11 08:05:31 +00:00
|
|
|
* @cfg {string} [icon=''] Symbolic name of an icon, e.g. "puzzle" or "wikiText"
|
2021-07-13 09:00:57 +00:00
|
|
|
* @cfg {string} label
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget = function VeUiMWTransclusionOutlineButtonWidget( config ) {
|
|
|
|
// Parent constructor
|
2021-09-10 19:18:59 +00:00
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.super.call( this, ve.extendObject( config, {
|
|
|
|
classes: [ 've-ui-mwTransclusionOutlineButtonWidget' ]
|
|
|
|
} ) );
|
2021-07-13 09:00:57 +00:00
|
|
|
|
|
|
|
// Mixin constructors
|
2021-07-13 10:15:47 +00:00
|
|
|
OO.ui.mixin.ButtonElement.call( this, {
|
2021-10-22 14:00:19 +00:00
|
|
|
// FIXME semantically this could be a <legend> and the surrounding OutlinePartWidget a <fieldset>
|
|
|
|
$button: $( '<span>' ),
|
2021-07-13 10:15:47 +00:00
|
|
|
framed: false
|
|
|
|
} );
|
2021-07-13 09:00:57 +00:00
|
|
|
OO.ui.mixin.IconElement.call( this, config );
|
2021-07-13 19:01:30 +00:00
|
|
|
OO.ui.mixin.TabIndexedElement.call( this, ve.extendObject( {
|
|
|
|
$tabIndexed: this.$button
|
|
|
|
}, config ) );
|
2021-07-13 09:00:57 +00:00
|
|
|
|
2021-10-27 09:58:51 +00:00
|
|
|
// FIXME hack for screen readers to understand the selection state
|
|
|
|
this.$button.attr( {
|
|
|
|
role: 'gridcell',
|
2021-10-27 09:39:03 +00:00
|
|
|
'aria-label': config.label,
|
2021-10-27 09:58:51 +00:00
|
|
|
'aria-selected': 'false'
|
|
|
|
} );
|
|
|
|
|
2021-07-13 09:00:57 +00:00
|
|
|
this.$element
|
2021-07-13 10:15:47 +00:00
|
|
|
.append( this.$button.append( this.$icon, this.$label ) );
|
2021-07-13 09:00:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Inheritance */
|
|
|
|
|
2021-08-27 16:22:37 +00:00
|
|
|
OO.inheritClass( ve.ui.MWTransclusionOutlineButtonWidget, OO.ui.OptionWidget );
|
2021-07-13 10:15:47 +00:00
|
|
|
OO.mixinClass( ve.ui.MWTransclusionOutlineButtonWidget, OO.ui.mixin.ButtonElement );
|
2021-07-13 09:00:57 +00:00
|
|
|
OO.mixinClass( ve.ui.MWTransclusionOutlineButtonWidget, OO.ui.mixin.IconElement );
|
2021-07-13 19:01:30 +00:00
|
|
|
OO.mixinClass( ve.ui.MWTransclusionOutlineButtonWidget, OO.ui.mixin.TabIndexedElement );
|
2021-08-27 16:22:37 +00:00
|
|
|
|
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.static.highlightable = false;
|
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.static.pressable = false;
|
2021-09-02 16:09:30 +00:00
|
|
|
|
|
|
|
/* Events */
|
|
|
|
|
|
|
|
/**
|
2021-09-22 09:49:44 +00:00
|
|
|
* @event keyPressed
|
|
|
|
* @param {number} key Typically one of the {@see OO.ui.Keys} constants
|
2021-09-02 16:09:30 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc OO.ui.mixin.ButtonElement
|
2021-09-22 09:49:44 +00:00
|
|
|
* @param {jQuery.Event} e
|
|
|
|
* @fires keyPressed
|
2021-09-02 16:09:30 +00:00
|
|
|
*/
|
2021-09-22 09:49:44 +00:00
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.prototype.onKeyDown = function ( e ) {
|
2021-10-28 09:51:18 +00:00
|
|
|
var isMac = ve.getSystemPlatform() === 'mac';
|
|
|
|
var withMetaKey = isMac ? e.metaKey : e.ctrlKey;
|
2021-09-22 09:49:44 +00:00
|
|
|
|
|
|
|
if ( e.which === OO.ui.Keys.SPACE &&
|
|
|
|
!withMetaKey && !e.shiftKey && !e.altKey
|
|
|
|
) {
|
2021-09-02 16:09:30 +00:00
|
|
|
// We know we can only select another part, so don't even try to unselect this one
|
|
|
|
if ( !this.isSelected() ) {
|
2021-09-22 09:49:44 +00:00
|
|
|
this.emit( 'keyPressed', e.which );
|
2021-09-02 16:09:30 +00:00
|
|
|
}
|
2021-09-22 09:49:44 +00:00
|
|
|
// The default behavior of pressing space is to scroll down
|
|
|
|
e.preventDefault();
|
|
|
|
return;
|
|
|
|
} else if ( ( e.which === OO.ui.Keys.UP || e.which === OO.ui.Keys.DOWN ) &&
|
|
|
|
withMetaKey && e.shiftKey &&
|
|
|
|
!e.altKey
|
|
|
|
) {
|
|
|
|
this.emit( 'keyPressed', e.which );
|
|
|
|
// TODO: Do we need e.preventDefault() and/or e.stopPropagation() here?
|
|
|
|
return;
|
2021-10-28 10:00:03 +00:00
|
|
|
} else if ( ( e.which === OO.ui.Keys.DELETE || ( isMac && e.which === OO.ui.Keys.BACKSPACE ) ) &&
|
2021-09-22 09:49:44 +00:00
|
|
|
withMetaKey &&
|
|
|
|
!e.shiftKey && !e.altKey
|
|
|
|
) {
|
2021-10-28 10:00:03 +00:00
|
|
|
this.emit( 'keyPressed', OO.ui.Keys.DELETE );
|
2021-09-22 09:49:44 +00:00
|
|
|
// To not trigger the "clear cache" feature in Chrome we must do both
|
2021-09-02 16:09:30 +00:00
|
|
|
e.preventDefault();
|
2021-09-22 09:49:44 +00:00
|
|
|
e.stopPropagation();
|
2021-09-02 16:09:30 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-22 09:49:44 +00:00
|
|
|
return OO.ui.mixin.ButtonElement.prototype.onKeyDown.call( this, e );
|
2021-09-02 16:09:30 +00:00
|
|
|
};
|
2021-10-27 09:58:51 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritDoc
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.prototype.setSelected = function ( state ) {
|
|
|
|
if ( this.$button ) {
|
|
|
|
this.$button.attr( 'aria-selected', state.toString() );
|
|
|
|
}
|
|
|
|
return ve.ui.MWTransclusionOutlineButtonWidget.super.prototype.setSelected.call( this, state );
|
|
|
|
};
|
2021-11-15 10:01:52 +00:00
|
|
|
|
|
|
|
/* Methods */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {jQuery} $description
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionOutlineButtonWidget.prototype.setAriaDescribedBy = function ( $description ) {
|
|
|
|
this.$button.attr( 'aria-describedby', $description.attr( 'id' ) );
|
|
|
|
};
|