2017-06-09 13:45:59 +00:00
|
|
|
/*!
|
|
|
|
* VisualEditor MediaWiki DiffPage init.
|
|
|
|
*
|
2023-12-01 16:06:11 +00:00
|
|
|
* @copyright See AUTHORS.txt
|
2017-06-09 13:45:59 +00:00
|
|
|
* @license The MIT License (MIT); see LICENSE.txt
|
|
|
|
*/
|
|
|
|
|
2019-02-20 20:23:43 +00:00
|
|
|
/* eslint-disable no-jquery/no-global-selector */
|
2019-01-08 17:00:09 +00:00
|
|
|
|
2017-06-09 13:45:59 +00:00
|
|
|
( function () {
|
2024-05-21 14:22:56 +00:00
|
|
|
let reviewModeButtonSelect, lastDiff,
|
2018-04-30 17:21:28 +00:00
|
|
|
$wikitextDiffContainer, $wikitextDiffHeader, $wikitextDiffBody,
|
2018-03-04 14:45:33 +00:00
|
|
|
$visualDiffContainer = $( '<div>' ),
|
|
|
|
$visualDiff = $( '<div>' ),
|
|
|
|
progress = new OO.ui.ProgressBarWidget( { classes: [ 've-init-mw-diffPage-loading' ] } ),
|
2022-12-12 19:49:26 +00:00
|
|
|
originalUrl = new URL( location.href ),
|
|
|
|
initMode = originalUrl.searchParams.get( 'diffmode' ) || mw.user.options.get( 'visualeditor-diffmode-historical' ) || 'source',
|
2017-06-09 13:45:59 +00:00
|
|
|
conf = mw.config.get( 'wgVisualEditorConfig' ),
|
2023-05-02 05:17:10 +00:00
|
|
|
pluginModules = conf.pluginModules.filter( mw.loader.getState ),
|
|
|
|
diffTypeSwitch;
|
2017-06-09 13:45:59 +00:00
|
|
|
|
2022-01-06 16:47:35 +00:00
|
|
|
if ( initMode !== 'visual' ) {
|
2018-02-22 17:48:37 +00:00
|
|
|
// Enforce a valid mode, to avoid visual glitches in button-selection.
|
2022-01-06 16:47:35 +00:00
|
|
|
initMode = 'source';
|
2018-02-22 17:48:37 +00:00
|
|
|
}
|
2024-05-21 14:22:56 +00:00
|
|
|
let mode = initMode;
|
2018-02-22 17:48:37 +00:00
|
|
|
|
2018-03-04 14:45:33 +00:00
|
|
|
$visualDiffContainer.append(
|
|
|
|
progress.$element.addClass( 'oo-ui-element-hidden' ),
|
|
|
|
$visualDiff
|
|
|
|
);
|
|
|
|
|
2017-06-09 13:45:59 +00:00
|
|
|
function onReviewModeButtonSelectSelect( item ) {
|
2024-05-21 14:22:56 +00:00
|
|
|
let oldPageName, newPageName;
|
2017-07-27 12:26:55 +00:00
|
|
|
if ( mw.config.get( 'wgCanonicalSpecialPageName' ) !== 'ComparePages' ) {
|
|
|
|
oldPageName = newPageName = mw.config.get( 'wgRelevantPageName' );
|
|
|
|
} else {
|
2024-05-21 14:22:56 +00:00
|
|
|
const params = new URLSearchParams( location.search );
|
2023-04-13 01:42:18 +00:00
|
|
|
oldPageName = params.get( 'page1' );
|
|
|
|
newPageName = params.get( 'page2' );
|
2017-07-27 12:26:55 +00:00
|
|
|
}
|
|
|
|
|
2023-05-02 05:17:10 +00:00
|
|
|
mode = item.getData();
|
2024-05-21 14:22:56 +00:00
|
|
|
const isVisual = mode === 'visual';
|
2018-03-04 14:45:33 +00:00
|
|
|
|
2023-06-26 11:19:08 +00:00
|
|
|
$visualDiffContainer.toggleClass( 'oo-ui-element-hidden', !isVisual );
|
|
|
|
$wikitextDiffBody.toggleClass( 'oo-ui-element-hidden', isVisual );
|
2023-05-02 05:17:10 +00:00
|
|
|
|
2023-09-22 23:14:22 +00:00
|
|
|
// If inline switch exists
|
2023-05-02 05:17:10 +00:00
|
|
|
if ( typeof diffTypeSwitch !== 'undefined' ) {
|
|
|
|
diffTypeSwitch.setDisabled( isVisual );
|
|
|
|
}
|
|
|
|
|
2024-05-21 14:22:56 +00:00
|
|
|
const $revSlider = $( '.mw-revslider-container' );
|
2018-03-04 14:45:33 +00:00
|
|
|
$revSlider.toggleClass( 've-init-mw-diffPage-revSlider-visual', isVisual );
|
2018-04-30 17:21:28 +00:00
|
|
|
if ( isVisual ) {
|
|
|
|
// Highlight the headers using the same styles as the diff, to better indicate
|
|
|
|
// the meaning of headers when not using two-column diff.
|
|
|
|
$wikitextDiffHeader.find( '#mw-diff-otitle1' ).attr( 'data-diff-action', 'remove' );
|
|
|
|
$wikitextDiffHeader.find( '#mw-diff-ntitle1' ).attr( 'data-diff-action', 'insert' );
|
|
|
|
} else {
|
|
|
|
$wikitextDiffHeader.find( '#mw-diff-otitle1' ).removeAttr( 'data-diff-action' );
|
|
|
|
$wikitextDiffHeader.find( '#mw-diff-ntitle1' ).removeAttr( 'data-diff-action' );
|
|
|
|
}
|
2017-06-09 13:45:59 +00:00
|
|
|
|
2024-05-21 14:22:56 +00:00
|
|
|
const oldId = mw.config.get( 'wgDiffOldId' );
|
|
|
|
const newId = mw.config.get( 'wgDiffNewId' );
|
2018-03-04 14:45:33 +00:00
|
|
|
if ( isVisual && !(
|
|
|
|
lastDiff && lastDiff.oldId === oldId && lastDiff.newId === newId &&
|
|
|
|
lastDiff.oldPageName === oldPageName && lastDiff.newPageName === newPageName
|
|
|
|
) ) {
|
|
|
|
$visualDiff.empty();
|
|
|
|
progress.$element.removeClass( 'oo-ui-element-hidden' );
|
2017-06-09 13:45:59 +00:00
|
|
|
// TODO: Load a smaller subset of VE for computing the visual diff
|
2024-05-21 14:22:56 +00:00
|
|
|
const modulePromise = mw.loader.using( [ 'ext.visualEditor.articleTarget', 'ext.visualEditor.mwmeta' ].concat( pluginModules ) );
|
2024-04-30 16:44:25 +00:00
|
|
|
mw.libs.ve.diffLoader.getVisualDiffGeneratorPromise( oldId, newId, modulePromise, oldPageName, newPageName ).then( ( visualDiffGenerator ) => {
|
2019-07-22 20:33:15 +00:00
|
|
|
// This class is loaded via modulePromise above
|
|
|
|
// eslint-disable-next-line no-undef
|
2024-05-21 14:22:56 +00:00
|
|
|
const diffElement = new ve.ui.DiffElement( visualDiffGenerator(), { classes: [ 've-init-mw-diffPage-diff' ] } );
|
2021-10-14 16:26:03 +00:00
|
|
|
diffElement.$document.addClass( 'mw-parser-output content' );
|
2017-06-09 13:45:59 +00:00
|
|
|
|
2022-01-19 01:26:14 +00:00
|
|
|
mw.libs.ve.fixFragmentLinks( diffElement.$document[ 0 ], mw.Title.newFromText( newPageName ), 'mw-diffpage-visualdiff-' );
|
|
|
|
|
2018-03-04 14:45:33 +00:00
|
|
|
progress.$element.addClass( 'oo-ui-element-hidden' );
|
|
|
|
$visualDiff.append( diffElement.$element );
|
|
|
|
lastDiff = {
|
|
|
|
oldId: oldId,
|
|
|
|
newId: newId,
|
|
|
|
oldPageName: oldPageName,
|
|
|
|
newPageName: newPageName
|
|
|
|
};
|
2017-06-09 13:45:59 +00:00
|
|
|
|
|
|
|
diffElement.positionDescriptions();
|
2024-04-30 16:44:25 +00:00
|
|
|
}, ( code, data ) => {
|
2021-02-23 00:21:55 +00:00
|
|
|
mw.notify( new mw.Api().getErrorMessage( data ), { type: 'error' } );
|
|
|
|
reviewModeButtonSelect.selectItemByData( 'source' );
|
2024-04-30 16:44:25 +00:00
|
|
|
} ).catch( ( error ) => {
|
2022-11-30 01:23:45 +00:00
|
|
|
mw.notify( error.message, { type: 'error' } );
|
|
|
|
reviewModeButtonSelect.selectItemByData( 'source' );
|
|
|
|
throw error;
|
2017-06-09 13:45:59 +00:00
|
|
|
} );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-06 16:48:02 +00:00
|
|
|
function onReviewModeButtonSelectChoose( item ) {
|
2023-05-02 05:17:10 +00:00
|
|
|
mode = item.getData();
|
2022-01-06 16:48:02 +00:00
|
|
|
if ( mode !== mw.user.options.get( 'visualeditor-diffmode-historical' ) ) {
|
|
|
|
mw.user.options.set( 'visualeditor-diffmode-historical', mode );
|
|
|
|
// Same as ve.init.target.getLocalApi()
|
|
|
|
new mw.Api().saveOption( 'visualeditor-diffmode-historical', mode );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-30 16:44:25 +00:00
|
|
|
mw.hook( 'wikipage.diff' ).add( () => {
|
2023-06-10 14:44:18 +00:00
|
|
|
if ( mw.config.get( 'wgDiffOldId' ) === false || mw.config.get( 'wgDiffNewId' ) === false ) {
|
|
|
|
// Don't offer visual diffs for "fake" diffs where the revision to compare to is not given,
|
|
|
|
// e.g. when viewing a "diff" of page creation (T338388)
|
|
|
|
return;
|
|
|
|
}
|
2023-09-14 12:20:12 +00:00
|
|
|
if ( !Object.prototype.hasOwnProperty.call( conf.contentModels, mw.config.get( 'wgPageContentModel' ) ) ) {
|
|
|
|
// Don't offer visual diffs for non-wikitext pages.
|
|
|
|
// TODO: This is wrong if we're comparing revisions of two different pages, or if the content
|
|
|
|
// model of the page has changed between the revisions. The diff needs to give us more
|
|
|
|
// information about the revisions being compared… (T346252)
|
|
|
|
return;
|
|
|
|
}
|
2023-06-10 14:44:18 +00:00
|
|
|
|
2018-04-30 17:21:28 +00:00
|
|
|
$wikitextDiffContainer = $( 'table.diff[data-mw="interface"]' );
|
2018-05-08 19:57:59 +00:00
|
|
|
$wikitextDiffHeader = $wikitextDiffContainer.find( 'tr.diff-title' )
|
|
|
|
.add( $wikitextDiffContainer.find( 'td.diff-multi, td.diff-notice' ).parent() );
|
2018-04-30 17:21:28 +00:00
|
|
|
$wikitextDiffBody = $wikitextDiffContainer.find( 'tr' ).not( $wikitextDiffHeader );
|
|
|
|
$wikitextDiffContainer.after( $visualDiffContainer );
|
2018-03-04 15:01:26 +00:00
|
|
|
|
2017-06-09 13:45:59 +00:00
|
|
|
// The PHP widget was a ButtonGroupWidget, so replace with a
|
|
|
|
// ButtonSelectWidget instead of infusing.
|
|
|
|
reviewModeButtonSelect = new OO.ui.ButtonSelectWidget( {
|
|
|
|
items: [
|
|
|
|
new OO.ui.ButtonOptionWidget( { data: 'visual', icon: 'eye', label: mw.msg( 'visualeditor-savedialog-review-visual' ) } ),
|
|
|
|
new OO.ui.ButtonOptionWidget( { data: 'source', icon: 'wikiText', label: mw.msg( 'visualeditor-savedialog-review-wikitext' ) } )
|
|
|
|
]
|
|
|
|
} );
|
2022-01-06 16:48:02 +00:00
|
|
|
// Choose is only emitted when the user interacts with the widget, whereas
|
|
|
|
// select is emitted even when the mode is set programmatically (e.g. on load)
|
2017-06-09 13:45:59 +00:00
|
|
|
reviewModeButtonSelect.on( 'select', onReviewModeButtonSelectSelect );
|
2022-01-06 16:48:02 +00:00
|
|
|
reviewModeButtonSelect.on( 'choose', onReviewModeButtonSelectChoose );
|
2017-06-09 13:45:59 +00:00
|
|
|
$( '.ve-init-mw-diffPage-diffMode' ).empty().append( reviewModeButtonSelect.$element );
|
2023-11-10 09:14:19 +00:00
|
|
|
reviewModeButtonSelect.selectItemByData( mode );
|
2017-06-09 13:45:59 +00:00
|
|
|
} );
|
2023-05-02 05:17:10 +00:00
|
|
|
|
2024-04-30 16:44:25 +00:00
|
|
|
mw.hook( 'wikipage.diff.wikitextBodyUpdate' ).add( ( $wikitextBody ) => {
|
2023-05-02 05:17:10 +00:00
|
|
|
$wikitextDiffBody = $wikitextBody;
|
|
|
|
} );
|
|
|
|
|
2024-04-30 16:44:25 +00:00
|
|
|
mw.hook( 'wikipage.diff.diffTypeSwitch' ).add( ( inlineToggleSwitch ) => {
|
2023-05-02 05:17:10 +00:00
|
|
|
diffTypeSwitch = inlineToggleSwitch;
|
|
|
|
diffTypeSwitch.setDisabled( mode === 'visual' );
|
|
|
|
} );
|
2017-06-09 13:45:59 +00:00
|
|
|
}() );
|