2018-09-26 12:49:10 +00:00
|
|
|
/*!
|
2013-06-11 19:16:04 +00:00
|
|
|
* VisualEditor user interface MWTransclusionDialog class.
|
|
|
|
*
|
2020-01-08 17:13:04 +00:00
|
|
|
* @copyright 2011-2020 VisualEditor Team and others; see AUTHORS.txt
|
2013-06-11 19:16:04 +00:00
|
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2021-06-03 16:25:15 +00:00
|
|
|
* Dialog for inserting and editing MediaWiki transclusions, i.e. a sequence of one or more template
|
|
|
|
* invocations that strictly belong to each other (e.g. because they are unbalanced), possibly
|
|
|
|
* mixed with raw wikitext snippets.
|
|
|
|
*
|
|
|
|
* Note the base class {@see ve.ui.MWTemplateDialog} alone does not allow to manage more than a
|
|
|
|
* single template invocation. Most of the code for this feature set is exclusive to this subclass.
|
2013-06-11 19:16:04 +00:00
|
|
|
*
|
|
|
|
* @class
|
2014-04-10 19:07:40 +00:00
|
|
|
* @extends ve.ui.MWTemplateDialog
|
2013-06-11 19:16:04 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
2013-09-25 10:21:09 +00:00
|
|
|
* @param {Object} [config] Configuration options
|
2013-06-11 19:16:04 +00:00
|
|
|
*/
|
2014-08-21 00:50:54 +00:00
|
|
|
ve.ui.MWTransclusionDialog = function VeUiMWTransclusionDialog( config ) {
|
2021-08-05 14:57:16 +00:00
|
|
|
var veConfig = mw.config.get( 'wgVisualEditorConfig' );
|
|
|
|
|
2013-06-11 19:16:04 +00:00
|
|
|
// Parent constructor
|
2014-08-21 00:50:54 +00:00
|
|
|
ve.ui.MWTransclusionDialog.super.call( this, config );
|
2013-07-03 01:30:10 +00:00
|
|
|
|
2013-06-11 19:16:04 +00:00
|
|
|
// Properties
|
2021-06-04 15:41:41 +00:00
|
|
|
this.isSidebarExpanded = null;
|
2021-05-03 10:40:02 +00:00
|
|
|
|
2021-05-07 14:03:25 +00:00
|
|
|
this.resetConfirmation = new OO.ui.FieldsetLayout( {
|
|
|
|
classes: [ 'oo-ui-processDialog-errors' ]
|
|
|
|
} );
|
|
|
|
this.resetConfirmationTitle = new OO.ui.LabelWidget( {
|
|
|
|
label: OO.ui.deferMsg( 'visualeditor-dialog-transclusion-reset-confirmation-title' )
|
|
|
|
} );
|
|
|
|
this.resetConfirmationMessage = new OO.ui.MessageWidget( {
|
|
|
|
label: OO.ui.deferMsg( 'visualeditor-dialog-transclusion-reset-confirmation-message' ),
|
|
|
|
type: 'error'
|
|
|
|
} );
|
|
|
|
this.resetConfirmationCancelButton = new OO.ui.ButtonWidget( {
|
|
|
|
label: OO.ui.deferMsg( 'visualeditor-dialog-transclusion-reset-confirmation-cancel' )
|
|
|
|
} );
|
|
|
|
this.resetConfirmationResetButton = new OO.ui.ButtonWidget( {
|
|
|
|
label: OO.ui.deferMsg( 'visualeditor-dialog-transclusion-reset-confirmation-reset' ),
|
|
|
|
flags: [ 'primary', 'destructive' ]
|
|
|
|
} );
|
|
|
|
|
|
|
|
// Events
|
|
|
|
this.resetConfirmationCancelButton.connect( this, {
|
|
|
|
click: 'resetConfirmationHide'
|
|
|
|
} );
|
|
|
|
this.resetConfirmationResetButton.connect( this, {
|
|
|
|
click: 'resetConfirmationReset'
|
|
|
|
} );
|
2021-09-30 11:07:53 +00:00
|
|
|
this.setupHotkeyTriggers();
|
2021-05-07 14:03:25 +00:00
|
|
|
|
|
|
|
// Initialization
|
|
|
|
this.resetConfirmation.addItems( [
|
|
|
|
new OO.ui.HorizontalLayout( {
|
|
|
|
items: [ this.resetConfirmationTitle ],
|
|
|
|
classes: [ 'oo-ui-processDialog-errors-title' ]
|
|
|
|
} ),
|
|
|
|
this.resetConfirmationMessage,
|
|
|
|
new OO.ui.HorizontalLayout( {
|
|
|
|
items: [ this.resetConfirmationCancelButton, this.resetConfirmationResetButton ],
|
|
|
|
classes: [ 'oo-ui-processDialog-errors-actions' ]
|
|
|
|
} )
|
|
|
|
] );
|
|
|
|
this.resetConfirmation.toggle( false );
|
|
|
|
this.$content.append( this.resetConfirmation.$element );
|
|
|
|
|
2021-08-05 14:57:16 +00:00
|
|
|
// Temporary feature flags
|
|
|
|
this.useInlineDescriptions = veConfig.transclusionDialogInlineDescriptions;
|
|
|
|
this.useBackButton = veConfig.transclusionDialogBackButton;
|
|
|
|
this.useSearchImprovements = veConfig.templateSearchImprovements;
|
2021-08-05 14:31:56 +00:00
|
|
|
this.useNewSidebar = veConfig.transclusionDialogNewSidebar;
|
2021-08-05 14:57:16 +00:00
|
|
|
|
|
|
|
if ( this.useInlineDescriptions ) {
|
2021-05-03 10:40:02 +00:00
|
|
|
this.$element.addClass( 've-ui-mwTransclusionDialog-bigger' );
|
|
|
|
}
|
2021-08-05 14:57:16 +00:00
|
|
|
if ( this.useSearchImprovements ) {
|
2021-05-10 12:56:03 +00:00
|
|
|
this.$element.addClass( 've-ui-mwTransclusionDialog-enhancedSearch' );
|
|
|
|
}
|
2021-08-05 14:31:56 +00:00
|
|
|
if ( this.useNewSidebar ) {
|
|
|
|
this.$element.addClass( 've-ui-mwTransclusionDialog-newSidebar' );
|
|
|
|
}
|
2021-09-27 11:16:33 +00:00
|
|
|
|
|
|
|
this.$element.on( 'keydown', this.onKeyDown.bind( this ) );
|
2013-06-11 19:16:04 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Inheritance */
|
|
|
|
|
2014-04-10 19:07:40 +00:00
|
|
|
OO.inheritClass( ve.ui.MWTransclusionDialog, ve.ui.MWTemplateDialog );
|
2013-07-03 01:30:10 +00:00
|
|
|
|
2013-06-11 19:16:04 +00:00
|
|
|
/* Static Properties */
|
|
|
|
|
2014-03-04 22:56:14 +00:00
|
|
|
ve.ui.MWTransclusionDialog.static.name = 'transclusion';
|
|
|
|
|
2014-07-14 21:32:49 +00:00
|
|
|
ve.ui.MWTransclusionDialog.static.actions = ve.ui.MWTemplateDialog.static.actions.concat( [
|
|
|
|
{
|
2014-08-22 20:50:48 +00:00
|
|
|
action: 'mode',
|
2017-06-14 18:34:24 +00:00
|
|
|
modes: [ 'edit', 'insert' ],
|
|
|
|
// HACK: Will be set later, but we want measurements to be accurate in the mean time, this
|
|
|
|
// will not be needed when T93290 is resolved
|
|
|
|
label: $( document.createTextNode( '\u00a0' ) )
|
2021-05-07 10:40:36 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
action: 'back',
|
|
|
|
label: OO.ui.deferMsg( 'visualeditor-dialog-action-goback' ),
|
|
|
|
modes: [ 'edit', 'insert' ],
|
|
|
|
flags: [ 'safe', 'back' ]
|
2014-07-14 21:32:49 +00:00
|
|
|
}
|
|
|
|
] );
|
|
|
|
|
2021-07-16 12:24:46 +00:00
|
|
|
/** @inheritdoc */
|
2014-04-10 19:07:40 +00:00
|
|
|
ve.ui.MWTransclusionDialog.static.bookletLayoutConfig = ve.extendObject(
|
|
|
|
{},
|
|
|
|
ve.ui.MWTemplateDialog.static.bookletLayoutConfig,
|
2014-08-22 20:50:48 +00:00
|
|
|
{ outlined: true, editable: true }
|
2014-04-10 19:07:40 +00:00
|
|
|
);
|
|
|
|
|
2021-07-23 15:07:39 +00:00
|
|
|
ve.ui.MWTransclusionDialog.static.smallScreenMaxWidth = 540;
|
|
|
|
|
2013-06-11 19:16:04 +00:00
|
|
|
/* Methods */
|
|
|
|
|
2014-03-04 22:56:14 +00:00
|
|
|
/**
|
|
|
|
* Handle outline controls move events.
|
|
|
|
*
|
2021-06-18 13:51:25 +00:00
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
* @param {number} places Number of places to move the selected item
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.onOutlineControlsMove = function ( places ) {
|
2021-10-08 07:37:01 +00:00
|
|
|
var part = this.transclusionModel.getPartFromId( this.findSelectedItemId() );
|
|
|
|
if ( !part ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var index = this.transclusionModel.getParts().indexOf( part );
|
|
|
|
// Move part to new location, and if dialog is loaded switch to new part page
|
|
|
|
var promise = this.transclusionModel.addPart( part, index + places );
|
|
|
|
if ( this.loaded && !this.preventReselection ) {
|
|
|
|
promise.done( this.focusPart.bind( this, part.getId() ) );
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle outline controls remove events.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.onOutlineControlsRemove = function () {
|
2021-10-08 07:37:01 +00:00
|
|
|
var itemId = this.findSelectedItemId(),
|
|
|
|
part = this.transclusionModel.getPartFromId( itemId );
|
|
|
|
// Check if the part is the actual template, or one of its parameters
|
|
|
|
// TODO: This applies to the old sidebar only and can be removed later
|
|
|
|
if ( part instanceof ve.dm.MWTemplateModel && itemId !== part.getId() ) {
|
|
|
|
var param = part.getParameterFromId( itemId );
|
|
|
|
if ( param ) {
|
|
|
|
param.remove();
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
2021-10-08 07:37:01 +00:00
|
|
|
} else if ( part instanceof ve.dm.MWTransclusionPartModel ) {
|
|
|
|
this.transclusionModel.removePart( part );
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2021-09-27 11:16:33 +00:00
|
|
|
* Create a new template part at the end of the transclusion.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
2021-09-27 11:16:33 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.addTemplatePlaceholder = function () {
|
2014-04-24 00:22:45 +00:00
|
|
|
this.addPart( new ve.dm.MWTemplatePlaceholderModel( this.transclusionModel ) );
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle add content button click events.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
2021-09-30 11:07:53 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.addContent = function () {
|
2021-06-04 10:16:58 +00:00
|
|
|
this.addPart( new ve.dm.MWTransclusionContentModel( this.transclusionModel ) );
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle add parameter button click events.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
2021-09-30 11:07:53 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.addParameter = function () {
|
2021-10-08 07:37:01 +00:00
|
|
|
var part = this.transclusionModel.getPartFromId( this.findSelectedItemId() );
|
2021-06-18 14:09:52 +00:00
|
|
|
if ( !( part instanceof ve.dm.MWTemplateModel ) ) {
|
|
|
|
return;
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
2021-06-18 14:09:52 +00:00
|
|
|
|
|
|
|
// TODO: Use a distinct class for placeholder model rather than
|
|
|
|
// these magical "empty" constants.
|
2021-10-14 11:43:35 +00:00
|
|
|
var placeholderParameter = new ve.dm.MWParameterModel( part );
|
|
|
|
part.addParameter( placeholderParameter );
|
|
|
|
this.focusPart( placeholderParameter.getId() );
|
2021-10-14 11:59:46 +00:00
|
|
|
|
|
|
|
if ( this.useInlineDescriptions ) {
|
|
|
|
this.autoExpandSidebar();
|
|
|
|
}
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle booklet layout page set events.
|
|
|
|
*
|
2021-06-18 13:51:25 +00:00
|
|
|
* @private
|
2014-03-04 22:56:14 +00:00
|
|
|
* @param {OO.ui.PageLayout} page Active page
|
|
|
|
*/
|
2021-09-16 08:52:59 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.onBookletLayoutSetPage = function ( page ) {
|
2021-10-06 13:37:02 +00:00
|
|
|
var isLastPlaceholder = page instanceof ve.ui.MWTemplatePlaceholderPage &&
|
2021-10-06 15:57:10 +00:00
|
|
|
this.transclusionModel.isSingleTemplate(),
|
2021-06-18 14:09:52 +00:00
|
|
|
acceptsNewParameters = page instanceof ve.ui.MWTemplatePage ||
|
2021-09-13 14:28:27 +00:00
|
|
|
page instanceof ve.ui.MWParameterPage;
|
2021-06-18 14:09:52 +00:00
|
|
|
|
|
|
|
this.addParameterButton.setDisabled( !acceptsNewParameters || this.isReadOnly() );
|
2021-09-13 14:28:27 +00:00
|
|
|
this.bookletLayout.getOutlineControls().removeButton.toggle( !isLastPlaceholder );
|
2021-09-17 14:38:37 +00:00
|
|
|
|
|
|
|
if ( this.pocSidebar ) {
|
2021-09-30 11:08:10 +00:00
|
|
|
this.pocSidebar.setSelectionByPageName( page.getName() );
|
2021-09-17 14:38:37 +00:00
|
|
|
}
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
2013-06-11 19:16:04 +00:00
|
|
|
/**
|
2014-04-10 19:07:40 +00:00
|
|
|
* @inheritdoc
|
2013-06-11 19:16:04 +00:00
|
|
|
*/
|
2013-12-02 20:10:55 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.onReplacePart = function ( removed, added ) {
|
2014-04-10 19:07:40 +00:00
|
|
|
ve.ui.MWTransclusionDialog.super.prototype.onReplacePart.call( this, removed, added );
|
2021-08-23 11:25:15 +00:00
|
|
|
var parts = this.transclusionModel.getParts();
|
2014-07-14 21:32:49 +00:00
|
|
|
|
2021-08-23 11:25:15 +00:00
|
|
|
if ( parts.length === 0 ) {
|
2014-10-23 18:37:50 +00:00
|
|
|
this.addParameterButton.setDisabled( true );
|
2015-03-19 22:54:48 +00:00
|
|
|
this.addPart( new ve.dm.MWTemplatePlaceholderModel( this.transclusionModel ) );
|
2021-10-08 15:15:05 +00:00
|
|
|
} else if ( parts.length > 1 ) {
|
|
|
|
this.bookletLayout.getOutlineControls().toggle( true );
|
|
|
|
this.$element.removeClass( 've-ui-mwTransclusionDialog-single-transclusion' );
|
2014-10-23 18:37:50 +00:00
|
|
|
}
|
|
|
|
|
2021-08-23 11:25:15 +00:00
|
|
|
this.multipartMessage.toggle( parts.length > 1 && this.useNewSidebar );
|
|
|
|
|
2021-10-13 15:28:34 +00:00
|
|
|
this.autoExpandSidebar();
|
2021-09-20 14:50:36 +00:00
|
|
|
this.updateModeActionState();
|
2021-05-07 10:40:36 +00:00
|
|
|
this.updateActionSet();
|
2013-06-11 19:16:04 +00:00
|
|
|
};
|
|
|
|
|
2021-09-27 11:16:33 +00:00
|
|
|
/**
|
|
|
|
* @private
|
|
|
|
* @param {string} key
|
|
|
|
* @return {string}
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.getHotkeyCombo = function ( key ) {
|
|
|
|
return ( ve.getSystemPlatform() === 'mac' ? 'meta+' : 'ctrl+' ) + 'shift+' + key;
|
|
|
|
};
|
|
|
|
|
2021-09-30 11:07:53 +00:00
|
|
|
/**
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.setupHotkeyTriggers = function () {
|
|
|
|
var context = this;
|
|
|
|
|
|
|
|
this.hotkeyTriggers = {};
|
2021-10-20 08:53:44 +00:00
|
|
|
this.hotkeyTriggers[ this.getHotkeyCombo( 'x' ) ] = function () {
|
2021-09-30 11:07:53 +00:00
|
|
|
context.addTemplatePlaceholder();
|
|
|
|
};
|
|
|
|
this.hotkeyTriggers[ this.getHotkeyCombo( 'y' ) ] = function () {
|
|
|
|
context.addContent();
|
|
|
|
};
|
2021-10-20 08:53:44 +00:00
|
|
|
this.hotkeyTriggers[ this.getHotkeyCombo( 'd' ) ] = function () {
|
2021-09-30 11:07:53 +00:00
|
|
|
context.addParameter();
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-09-27 11:16:33 +00:00
|
|
|
/**
|
|
|
|
* Handles key down events.
|
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @param {jQuery.Event} e Key down event
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.onKeyDown = function ( e ) {
|
2021-09-30 11:07:53 +00:00
|
|
|
var eventTrigger = new ve.ui.Trigger( e ).toString();
|
2021-09-27 11:16:33 +00:00
|
|
|
|
2021-09-30 11:07:53 +00:00
|
|
|
if ( this.hotkeyTriggers[ eventTrigger ] ) {
|
|
|
|
this.hotkeyTriggers[ eventTrigger ]();
|
2021-09-27 11:16:33 +00:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-08-27 17:07:17 +00:00
|
|
|
/**
|
2021-10-08 07:37:01 +00:00
|
|
|
* @return {string|undefined} Any id, including slash-delimited template parameter ids
|
2021-08-27 17:07:17 +00:00
|
|
|
*/
|
2021-10-08 07:37:01 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.findSelectedItemId = function () {
|
2021-08-27 17:07:17 +00:00
|
|
|
if ( this.pocSidebar ) {
|
2021-10-08 07:37:01 +00:00
|
|
|
// TODO: This can't return parameter ids any more when the old sidebar is gone
|
2021-08-27 17:07:17 +00:00
|
|
|
return this.pocSidebar.findSelectedPartId();
|
|
|
|
}
|
|
|
|
|
|
|
|
var item = this.bookletLayout.getOutline().findSelectedItem();
|
|
|
|
return item && item.getData();
|
|
|
|
};
|
|
|
|
|
2014-03-06 23:39:40 +00:00
|
|
|
/**
|
2014-04-10 19:07:40 +00:00
|
|
|
* @inheritdoc
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
2014-04-10 19:07:40 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.getPageFromPart = function ( part ) {
|
|
|
|
var page = ve.ui.MWTransclusionDialog.super.prototype.getPageFromPart.call( this, part );
|
|
|
|
if ( !page && part instanceof ve.dm.MWTransclusionContentModel ) {
|
2019-02-24 12:38:03 +00:00
|
|
|
return new ve.ui.MWTransclusionContentPage( part, part.getId(), { $overlay: this.$overlay, isReadOnly: this.isReadOnly() } );
|
2014-04-10 19:07:40 +00:00
|
|
|
}
|
|
|
|
return page;
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2021-08-06 10:28:59 +00:00
|
|
|
* Automatically expand or collapse the sidebar according to default logic.
|
2014-03-04 22:56:14 +00:00
|
|
|
*
|
2021-09-20 09:43:18 +00:00
|
|
|
* @protected
|
2014-03-04 22:56:14 +00:00
|
|
|
*/
|
2021-08-06 10:28:59 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.autoExpandSidebar = function () {
|
2021-09-09 10:26:40 +00:00
|
|
|
var expandSidebar;
|
2021-08-06 10:28:59 +00:00
|
|
|
|
|
|
|
if ( this.useInlineDescriptions ) {
|
2021-09-20 13:49:44 +00:00
|
|
|
var isSmallScreen = this.isNarrowScreen();
|
2021-10-28 08:42:44 +00:00
|
|
|
|
|
|
|
var showOtherActions = isSmallScreen ||
|
|
|
|
this.actions.getOthers().some( function ( action ) {
|
|
|
|
// Check for unknown actions, show the toolbar if any are available.
|
|
|
|
return action.action !== 'mode';
|
|
|
|
} );
|
|
|
|
this.actions.forEach( { actions: [ 'mode' ] }, function ( action ) {
|
|
|
|
action.toggle( isSmallScreen );
|
|
|
|
} );
|
|
|
|
this.$otherActions.toggleClass( 'oo-ui-element-hidden', !showOtherActions );
|
|
|
|
|
2021-10-26 08:34:26 +00:00
|
|
|
if ( isSmallScreen && this.transclusionModel.isEmpty() ) {
|
2021-10-13 15:28:34 +00:00
|
|
|
expandSidebar = false;
|
|
|
|
} else if ( isSmallScreen &&
|
2021-10-07 18:45:13 +00:00
|
|
|
// eslint-disable-next-line no-jquery/no-class-state
|
|
|
|
this.$content.hasClass( 've-ui-mwTransclusionDialog-small-screen' )
|
|
|
|
) {
|
|
|
|
// We did this already. If the sidebar is visible or not is now the user's decision.
|
|
|
|
return;
|
2021-10-13 15:28:34 +00:00
|
|
|
} else {
|
|
|
|
expandSidebar = !isSmallScreen;
|
2021-10-07 18:45:13 +00:00
|
|
|
}
|
2021-10-27 10:19:59 +00:00
|
|
|
|
2021-08-06 10:28:59 +00:00
|
|
|
this.$content.toggleClass( 've-ui-mwTransclusionDialog-small-screen', isSmallScreen );
|
|
|
|
} else {
|
2021-10-06 15:57:10 +00:00
|
|
|
expandSidebar = !this.transclusionModel.isSingleTemplate();
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
2014-11-08 00:41:03 +00:00
|
|
|
|
2021-09-09 10:26:40 +00:00
|
|
|
this.toggleSidebar( expandSidebar );
|
2021-08-06 10:28:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set if the sidebar is visible (which means the dialog is expanded), or collapsed.
|
|
|
|
*
|
2021-09-09 10:26:40 +00:00
|
|
|
* @param {boolean} expandSidebar
|
2021-08-06 10:28:59 +00:00
|
|
|
* @private
|
|
|
|
*/
|
2021-09-09 10:26:40 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.toggleSidebar = function ( expandSidebar ) {
|
|
|
|
if ( this.isSidebarExpanded === expandSidebar ) {
|
2021-08-06 10:28:59 +00:00
|
|
|
return;
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
2021-08-06 10:28:59 +00:00
|
|
|
|
2021-09-09 10:26:40 +00:00
|
|
|
this.isSidebarExpanded = expandSidebar;
|
2021-08-06 10:28:59 +00:00
|
|
|
this.$content
|
2021-09-09 10:26:40 +00:00
|
|
|
.toggleClass( 've-ui-mwTransclusionDialog-collapsed', !expandSidebar )
|
|
|
|
.toggleClass( 've-ui-mwTransclusionDialog-expanded', expandSidebar );
|
2021-09-09 09:22:56 +00:00
|
|
|
|
|
|
|
var dialogSizeSidebarExpanded = this.useInlineDescriptions ? 'larger' : 'large';
|
|
|
|
var dialogSizeSidebarCollapsed = this.useNewSidebar ? dialogSizeSidebarExpanded : 'medium';
|
2021-08-06 10:28:59 +00:00
|
|
|
this.ignoreNextWindowResizeEvent = true;
|
2021-09-09 10:26:40 +00:00
|
|
|
this.setSize( expandSidebar ? dialogSizeSidebarExpanded : dialogSizeSidebarCollapsed );
|
2021-09-09 09:22:56 +00:00
|
|
|
|
2021-09-09 10:26:40 +00:00
|
|
|
this.bookletLayout.toggleOutline( expandSidebar );
|
2021-08-06 10:28:59 +00:00
|
|
|
this.updateTitle();
|
|
|
|
this.updateModeActionState();
|
|
|
|
|
|
|
|
// HACK blur any active input so that its dropdown will be hidden and won't end
|
|
|
|
// up being mispositioned
|
|
|
|
this.$content.find( 'input:focus' ).trigger( 'blur' );
|
2021-09-20 13:49:44 +00:00
|
|
|
|
|
|
|
if ( this.useInlineDescriptions && this.pocSidebar && this.loaded && this.isNarrowScreen() ) {
|
|
|
|
// Reapply selection and scrolling when switching between panes.
|
|
|
|
// FIXME: decouple from descendants
|
|
|
|
var selectedPage = this.bookletLayout.stackLayout.getCurrentItem();
|
|
|
|
if ( selectedPage ) {
|
|
|
|
var name = selectedPage.getName();
|
|
|
|
var dialog = this;
|
|
|
|
// Align whichever panel is becoming visible, after animation completes.
|
|
|
|
// TODO: Should hook onto an animation promise—but is this possible when pure CSS?
|
|
|
|
setTimeout( function () {
|
|
|
|
if ( expandSidebar ) {
|
2021-09-30 11:08:10 +00:00
|
|
|
dialog.pocSidebar.setSelectionByPageName( name );
|
2021-09-20 13:49:44 +00:00
|
|
|
} else {
|
|
|
|
selectedPage.scrollElementIntoView();
|
|
|
|
// TODO: Find a reliable way to refocus.
|
|
|
|
// dialog.focusPart( name );
|
|
|
|
}
|
|
|
|
}, OO.ui.theme.getDialogTransitionDuration() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.isNarrowScreen = function () {
|
|
|
|
return $( window ).width() <= this.constructor.static.smallScreenMaxWidth;
|
2014-03-06 23:39:40 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2021-06-18 13:51:25 +00:00
|
|
|
* @inheritdoc
|
2014-03-06 23:39:40 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.updateTitle = function () {
|
2021-10-06 15:57:10 +00:00
|
|
|
if ( !this.transclusionModel.isSingleTemplate() ) {
|
2021-06-22 14:02:31 +00:00
|
|
|
this.title.setLabel( ve.msg( 'visualeditor-dialog-transclusion-title-edit-transclusion' ) );
|
2014-04-30 21:34:45 +00:00
|
|
|
} else {
|
|
|
|
// Parent method
|
|
|
|
ve.ui.MWTransclusionDialog.super.prototype.updateTitle.call( this );
|
|
|
|
}
|
2014-03-04 22:56:14 +00:00
|
|
|
};
|
|
|
|
|
2014-11-08 00:41:03 +00:00
|
|
|
/**
|
2017-06-14 15:52:18 +00:00
|
|
|
* Update the state of the 'mode' action
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2014-11-08 00:41:03 +00:00
|
|
|
*/
|
2017-06-14 15:52:18 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.updateModeActionState = function () {
|
2021-10-01 13:45:44 +00:00
|
|
|
var isExpanded = this.isSidebarExpanded,
|
2021-06-08 08:37:47 +00:00
|
|
|
label = ve.msg( isExpanded ?
|
2021-06-11 11:08:45 +00:00
|
|
|
'visualeditor-dialog-transclusion-collapse-options' :
|
|
|
|
'visualeditor-dialog-transclusion-expand-options' );
|
2021-06-08 08:37:47 +00:00
|
|
|
|
2014-11-08 00:41:03 +00:00
|
|
|
this.actions.forEach( { actions: [ 'mode' ] }, function ( action ) {
|
2020-03-19 13:39:53 +00:00
|
|
|
action.setLabel( label );
|
2021-06-08 08:37:47 +00:00
|
|
|
action.$button.attr( 'aria-expanded', isExpanded ? 1 : 0 );
|
2014-11-08 00:41:03 +00:00
|
|
|
} );
|
2017-06-14 15:52:18 +00:00
|
|
|
|
2021-10-01 13:45:44 +00:00
|
|
|
// Old sidebar: Only a single template can be collapsed, except it's still the initial
|
|
|
|
// placeholder.
|
|
|
|
// New sidebar: The button is only visible on very narrow screens, {@see autoExpandSidebar}.
|
|
|
|
// It's always needed, except in the initial placeholder state.
|
2021-10-06 15:57:10 +00:00
|
|
|
var isInitialState = !isExpanded && this.transclusionModel.isEmpty(),
|
|
|
|
canCollapse = ( this.transclusionModel.isSingleTemplate() || this.useInlineDescriptions ) &&
|
|
|
|
!isInitialState;
|
2021-10-01 13:45:44 +00:00
|
|
|
this.actions.setAbilities( { mode: canCollapse } );
|
2014-11-08 00:41:03 +00:00
|
|
|
};
|
|
|
|
|
2014-03-04 22:56:14 +00:00
|
|
|
/**
|
|
|
|
* Add a part to the transclusion.
|
|
|
|
*
|
|
|
|
* @param {ve.dm.MWTransclusionPartModel} part Part to add
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.addPart = function ( part ) {
|
2021-06-21 08:10:27 +00:00
|
|
|
var parts = this.transclusionModel.getParts(),
|
2021-10-08 07:37:01 +00:00
|
|
|
selectedPart = this.transclusionModel.getPartFromId( this.findSelectedItemId() );
|
|
|
|
// Insert after selected part, or at the end if nothing is selected
|
|
|
|
var index = selectedPart ? parts.indexOf( selectedPart ) + 1 : parts.length;
|
|
|
|
// Add the part, and if dialog is loaded switch to part page
|
|
|
|
var promise = this.transclusionModel.addPart( part, index );
|
|
|
|
if ( this.loaded && !this.preventReselection ) {
|
|
|
|
promise.done( this.focusPart.bind( this, part.getId() ) );
|
2014-03-04 22:56:14 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-07-14 21:32:49 +00:00
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.getActionProcess = function ( action ) {
|
2021-05-07 10:40:36 +00:00
|
|
|
if ( action === 'back' ) {
|
2021-09-27 09:16:14 +00:00
|
|
|
if ( !this.transclusionModel.containsValuableData() ) {
|
2021-05-07 14:03:25 +00:00
|
|
|
return this.getActionProcess( 'reset' );
|
|
|
|
}
|
|
|
|
return new OO.ui.Process( function () {
|
|
|
|
this.resetConfirmation.toggle( true );
|
|
|
|
}, this );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( action === 'reset' ) {
|
2021-05-07 10:40:36 +00:00
|
|
|
return new OO.ui.Process( function () {
|
|
|
|
this.resetDialog();
|
|
|
|
}, this );
|
|
|
|
}
|
|
|
|
|
2014-07-14 21:32:49 +00:00
|
|
|
if ( action === 'mode' ) {
|
|
|
|
return new OO.ui.Process( function () {
|
2021-06-08 08:37:47 +00:00
|
|
|
this.toggleSidebar( !this.isSidebarExpanded );
|
2014-07-14 21:32:49 +00:00
|
|
|
}, this );
|
|
|
|
}
|
|
|
|
|
|
|
|
return ve.ui.MWTransclusionDialog.super.prototype.getActionProcess.call( this, action );
|
|
|
|
};
|
|
|
|
|
2021-05-07 10:40:36 +00:00
|
|
|
/**
|
|
|
|
* Update the widgets in the dialog's action bar.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2021-05-07 10:40:36 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.updateActionSet = function () {
|
2021-08-05 14:57:16 +00:00
|
|
|
var backButton = this.actions.get( { flags: [ 'back' ] } ).pop(),
|
2021-06-02 09:59:38 +00:00
|
|
|
saveButton = this.actions.get( { actions: [ 'done' ] } ).pop();
|
|
|
|
|
|
|
|
if ( saveButton && this.getMode() === 'edit' ) {
|
|
|
|
saveButton.setLabel( ve.msg( 'visualeditor-dialog-transclusion-action-save' ) );
|
|
|
|
}
|
2021-05-07 10:40:36 +00:00
|
|
|
|
2021-08-05 14:57:16 +00:00
|
|
|
if ( this.useBackButton ) {
|
2021-05-07 10:40:36 +00:00
|
|
|
var closeButton = this.actions.get( { flags: [ 'close' ] } ).pop(),
|
2021-10-06 15:57:10 +00:00
|
|
|
canGoBack = this.getMode() === 'insert' && !this.transclusionModel.isEmpty();
|
2021-05-07 10:40:36 +00:00
|
|
|
|
2021-10-06 15:57:10 +00:00
|
|
|
closeButton.toggle( !canGoBack );
|
|
|
|
backButton.toggle( canGoBack );
|
2021-05-07 10:40:36 +00:00
|
|
|
} else {
|
2021-09-08 13:33:10 +00:00
|
|
|
backButton.toggle( false );
|
2021-05-07 10:40:36 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-05-07 14:03:25 +00:00
|
|
|
/**
|
|
|
|
* Dismisses the reset confirmation.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2021-05-07 14:03:25 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.resetConfirmationHide = function () {
|
|
|
|
this.resetConfirmation.toggle( false );
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dismisses the reset confirmation and runs the reset action.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2021-05-07 14:03:25 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.resetConfirmationReset = function () {
|
|
|
|
this.resetConfirmationHide();
|
|
|
|
this.executeAction( 'reset' );
|
|
|
|
};
|
|
|
|
|
2021-05-07 10:40:36 +00:00
|
|
|
/**
|
|
|
|
* Revert the dialog back to its initial state.
|
2021-06-18 13:51:25 +00:00
|
|
|
*
|
|
|
|
* @private
|
2021-05-07 10:40:36 +00:00
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.resetDialog = function () {
|
|
|
|
var target = this;
|
|
|
|
this.transclusionModel.reset();
|
2021-07-16 12:54:09 +00:00
|
|
|
if ( this.pocSidebar ) {
|
|
|
|
this.pocSidebar.clear();
|
|
|
|
}
|
2021-05-07 10:40:36 +00:00
|
|
|
this.bookletLayout.clearPages();
|
|
|
|
this.transclusionModel
|
|
|
|
.addPart( new ve.dm.MWTemplatePlaceholderModel( this.transclusionModel ), 0 )
|
|
|
|
.done( function () {
|
2021-08-06 10:28:59 +00:00
|
|
|
target.autoExpandSidebar();
|
2021-05-07 10:40:36 +00:00
|
|
|
} );
|
|
|
|
};
|
|
|
|
|
2013-11-05 00:29:50 +00:00
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.initialize = function () {
|
|
|
|
// Parent method
|
2014-04-10 19:07:40 +00:00
|
|
|
ve.ui.MWTransclusionDialog.super.prototype.initialize.call( this );
|
2013-11-05 00:29:50 +00:00
|
|
|
|
|
|
|
// Properties
|
2014-03-04 22:56:14 +00:00
|
|
|
this.addTemplateButton = new OO.ui.ButtonWidget( {
|
2014-08-22 20:50:48 +00:00
|
|
|
framed: false,
|
2017-07-06 16:52:56 +00:00
|
|
|
icon: 'puzzle',
|
2021-05-11 10:49:06 +00:00
|
|
|
title: ve.msg( 'visualeditor-dialog-transclusion-add-template' )
|
2014-03-04 22:56:14 +00:00
|
|
|
} );
|
|
|
|
this.addContentButton = new OO.ui.ButtonWidget( {
|
2014-08-22 20:50:48 +00:00
|
|
|
framed: false,
|
2015-11-04 14:15:29 +00:00
|
|
|
icon: 'wikiText',
|
2021-08-05 14:31:56 +00:00
|
|
|
title: ve.msg( this.useNewSidebar ?
|
|
|
|
'visualeditor-dialog-transclusion-add-wikitext' :
|
|
|
|
'visualeditor-dialog-transclusion-add-content' )
|
2014-03-04 22:56:14 +00:00
|
|
|
} );
|
|
|
|
this.addParameterButton = new OO.ui.ButtonWidget( {
|
2014-08-22 20:50:48 +00:00
|
|
|
framed: false,
|
|
|
|
icon: 'parameter',
|
|
|
|
title: ve.msg( 'visualeditor-dialog-transclusion-add-param' )
|
2014-03-04 22:56:14 +00:00
|
|
|
} );
|
2013-11-05 00:29:50 +00:00
|
|
|
|
2021-08-05 14:31:56 +00:00
|
|
|
this.bookletLayout.getOutlineControls().addItems( [ this.addTemplateButton, this.addContentButton ] );
|
|
|
|
if ( !this.useNewSidebar ) {
|
|
|
|
this.bookletLayout.getOutlineControls().addItems( [ this.addParameterButton ] );
|
|
|
|
}
|
|
|
|
|
2021-08-23 11:25:15 +00:00
|
|
|
this.multipartMessage = new OO.ui.MessageWidget( {
|
|
|
|
label: ve.msg( 'visualeditor-dialog-transclusion-multipart-message' ),
|
|
|
|
classes: [ 've-ui-mwTransclusionDialog-multipart-message' ]
|
|
|
|
} );
|
|
|
|
|
2013-11-05 00:29:50 +00:00
|
|
|
// Events
|
2021-07-23 15:07:39 +00:00
|
|
|
if ( this.useInlineDescriptions ) {
|
|
|
|
this.getManager().connect( this, { resize: ve.debounce( this.onWindowResize.bind( this ) ) } );
|
|
|
|
}
|
2021-09-16 08:52:59 +00:00
|
|
|
this.bookletLayout.connect( this, { set: 'onBookletLayoutSetPage' } );
|
2021-08-05 14:31:56 +00:00
|
|
|
this.bookletLayout.$menu.find( '[ role="listbox" ]' ).first()
|
|
|
|
.attr( 'aria-label', ve.msg( 'visualeditor-dialog-transclusion-templates-menu-aria-label' ) );
|
2021-09-27 11:16:33 +00:00
|
|
|
this.addTemplateButton.connect( this, { click: 'addTemplatePlaceholder' } );
|
2021-09-30 11:07:53 +00:00
|
|
|
this.addContentButton.connect( this, { click: 'addContent' } );
|
|
|
|
this.addParameterButton.connect( this, { click: 'addParameter' } );
|
2021-08-05 14:31:56 +00:00
|
|
|
this.bookletLayout.getOutlineControls().connect( this, {
|
|
|
|
move: 'onOutlineControlsMove',
|
|
|
|
remove: 'onOutlineControlsRemove'
|
|
|
|
} );
|
|
|
|
if ( this.useNewSidebar ) {
|
2021-08-23 11:25:15 +00:00
|
|
|
this.bookletLayout.stackLayout.$element.prepend( this.multipartMessage.$element );
|
2021-08-05 14:31:56 +00:00
|
|
|
}
|
2014-04-17 22:03:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
2014-05-31 04:47:08 +00:00
|
|
|
ve.ui.MWTransclusionDialog.prototype.getSetupProcess = function ( data ) {
|
|
|
|
return ve.ui.MWTransclusionDialog.super.prototype.getSetupProcess.call( this, data )
|
2014-07-14 21:32:49 +00:00
|
|
|
.next( function () {
|
2019-02-24 12:38:03 +00:00
|
|
|
var isReadOnly = this.isReadOnly();
|
|
|
|
this.addTemplateButton.setDisabled( isReadOnly );
|
|
|
|
this.addContentButton.setDisabled( isReadOnly );
|
|
|
|
this.addParameterButton.setDisabled( isReadOnly );
|
|
|
|
this.bookletLayout.getOutlineControls().setAbilities( {
|
|
|
|
move: !isReadOnly,
|
|
|
|
remove: !isReadOnly
|
|
|
|
} );
|
|
|
|
|
2021-08-05 14:31:56 +00:00
|
|
|
if ( this.useNewSidebar ) {
|
2021-10-06 15:57:10 +00:00
|
|
|
this.bookletLayout.getOutlineControls().toggle( !this.transclusionModel.isSingleTemplate() );
|
2021-08-17 11:53:28 +00:00
|
|
|
this.$element.toggleClass(
|
|
|
|
've-ui-mwTransclusionDialog-single-transclusion',
|
2021-10-06 15:57:10 +00:00
|
|
|
this.transclusionModel.isSingleTemplate()
|
2021-08-17 11:53:28 +00:00
|
|
|
);
|
2021-08-05 14:31:56 +00:00
|
|
|
}
|
|
|
|
|
2017-06-14 15:52:18 +00:00
|
|
|
this.updateModeActionState();
|
2021-08-06 10:28:59 +00:00
|
|
|
this.autoExpandSidebar();
|
2014-05-31 04:47:08 +00:00
|
|
|
}, this );
|
2013-11-05 00:29:50 +00:00
|
|
|
};
|
|
|
|
|
2021-07-23 15:07:39 +00:00
|
|
|
/**
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.onWindowResize = function () {
|
|
|
|
if ( !this.ignoreNextWindowResizeEvent ) {
|
2021-08-06 10:28:59 +00:00
|
|
|
this.autoExpandSidebar();
|
2021-07-23 15:07:39 +00:00
|
|
|
}
|
|
|
|
this.ignoreNextWindowResizeEvent = false;
|
|
|
|
};
|
|
|
|
|
2021-05-03 10:40:02 +00:00
|
|
|
/**
|
|
|
|
* @inheritdoc
|
|
|
|
*
|
|
|
|
* Temporary override to increase dialog size when a feature flag is enabled.
|
|
|
|
*/
|
|
|
|
ve.ui.MWTransclusionDialog.prototype.getSizeProperties = function () {
|
2021-05-12 08:14:23 +00:00
|
|
|
var sizeProps = ve.ui.MWTransclusionDialog.super.prototype.getSizeProperties.call( this );
|
2021-05-03 10:40:02 +00:00
|
|
|
|
2021-08-05 14:57:16 +00:00
|
|
|
if ( this.useInlineDescriptions ) {
|
2021-05-12 08:14:23 +00:00
|
|
|
sizeProps = ve.extendObject( { height: '90%' }, sizeProps );
|
2021-05-03 10:40:02 +00:00
|
|
|
}
|
|
|
|
|
2021-05-12 08:14:23 +00:00
|
|
|
return sizeProps;
|
2021-05-03 10:40:02 +00:00
|
|
|
};
|
|
|
|
|
2014-03-04 22:56:14 +00:00
|
|
|
/* Registration */
|
|
|
|
|
2014-04-21 22:31:21 +00:00
|
|
|
ve.ui.windowFactory.register( ve.ui.MWTransclusionDialog );
|