/*!
 * JavaScript for diff inline toggle
 */
module.exports = function ( $inlineToggleSwitchLayout ) {
	const url = new URL( location.href ),
		api = new mw.Api(),
		$inlineLegendContainer = $( '.mw-diff-inline-legend' ),
		inlineToggleSwitchLayout = OO.ui.FieldLayout.static.infuse( $inlineToggleSwitchLayout ),
		inlineToggleSwitch = inlineToggleSwitchLayout.getField();

	inlineToggleSwitch.on( 'change', ( e ) => {
		onDiffTypeInlineChange( e, true );
	} );
	inlineToggleSwitch.on( 'disable', onDiffTypeInlineDisabled );

	const $wikitextDiffContainer = $( 'table.diff[data-mw="interface"]' );
	const $wikitextDiffHeader = $wikitextDiffContainer.find( 'tr.diff-title' )
		.add( $wikitextDiffContainer.find( 'td.diff-multi, td.diff-notice' ).parent() );
	let $wikitextDiffBody = $wikitextDiffContainer.find( 'tr' ).not( $wikitextDiffHeader );

	let $wikitextDiffBodyInline, $wikitextDiffBodyTable;
	if ( inlineToggleSwitch.getValue() ) {
		$wikitextDiffBodyInline = $wikitextDiffBody;
	} else {
		$wikitextDiffBodyTable = $wikitextDiffBody;
	}

	/**
	 * Handler for inlineToggleSwitch onChange event
	 *
	 * @param {boolean} isInline
	 * @param {boolean} saveDiffTypeOption
	 */
	function onDiffTypeInlineChange( isInline, saveDiffTypeOption ) {
		if ( ( isInline && typeof $wikitextDiffBodyInline === 'undefined' ) ||
			( !isInline && typeof $wikitextDiffBodyTable === 'undefined' ) ) {
			fetchDiff( isInline );
		} else {
			toggleDiffTypeVisibility( isInline );
		}

		if ( saveDiffTypeOption ) {
			api.saveOption( 'diff-type', isInline ? 'inline' : 'table' )
				.fail( ( error ) => {
					if ( error === 'notloggedin' ) {
						// Can't save preference, so use query parameter stickiness
						switchQueryParams( isInline );
					}
				} );
		}
	}

	/**
	 * Toggle legend and the DOM containers according to the format selected.
	 *
	 * @param {boolean} isInline
	 */
	function toggleDiffTypeVisibility( isInline ) {
		$inlineLegendContainer.toggleClass( 'oo-ui-element-hidden', !isInline );
		if ( typeof $wikitextDiffBodyInline !== 'undefined' ) {
			$wikitextDiffBodyInline.toggleClass( 'oo-ui-element-hidden', !isInline );
		}

		if ( typeof $wikitextDiffBodyTable !== 'undefined' ) {
			$wikitextDiffBodyTable.toggleClass( 'oo-ui-element-hidden', isInline );
		}
	}

	/**
	 * Change the query parameters to maintain the new diff type.
	 * This is used for anonymous users.
	 *
	 * @param {boolean} isInline
	 */
	function switchQueryParams( isInline ) {
		$( '#differences-prevlink, #differences-nextlink' )
			.each( function () {
				let linkUrl;
				try {
					linkUrl = new URL( this.href );
				} catch ( e ) {
					return;
				}
				if ( isInline ) {
					linkUrl.searchParams.set( 'diff-type', 'inline' );
				} else {
					linkUrl.searchParams.delete( 'diff-type' );
				}
				this.href = linkUrl.toString();
			} );
	}

	/**
	 * Toggle the legend when the toggle switch disabled state changes.
	 *
	 * @param {boolean} disabled
	 */
	function onDiffTypeInlineDisabled( disabled ) {
		if ( disabled ) {
			$inlineLegendContainer.addClass( 'oo-ui-element-hidden' );
		} else {
			$inlineLegendContainer.toggleClass( 'oo-ui-element-hidden', !inlineToggleSwitch.getValue() );
			// When Inline Switch is enabled, toggle wikitext according to its value.
			// Do not save user 'diff-type' preference
			onDiffTypeInlineChange( inlineToggleSwitch.getValue(), false );
		}
	}

	/**
	 * Fetch the diff through mw.API in the given format.
	 *
	 * @param {boolean} isInline
	 */
	function fetchDiff( isInline ) {
		const diffType = isInline ? 'inline' : 'table',
			oldRevId = mw.config.get( 'wgDiffOldId' ),
			newRevId = mw.config.get( 'wgDiffNewId' );

		let oldPageName, newPageName;
		if ( mw.config.get( 'wgCanonicalSpecialPageName' ) !== 'ComparePages' ) {
			oldPageName = newPageName = mw.config.get( 'wgRelevantPageName' );
		} else {
			oldPageName = url.searchParams.get( 'page1' );
			newPageName = url.searchParams.get( 'page2' );
		}

		const apiParams = {
			action: 'compare',
			fromtitle: oldPageName,
			totitle: newPageName,
			fromrev: oldRevId,
			torev: newRevId,
			difftype: diffType
		};

		api.get( apiParams ).done( ( diffData ) => {
			if ( isInline ) {
				$wikitextDiffBodyInline = $( diffData.compare[ '*' ] );
			} else {
				$wikitextDiffBodyTable = $( diffData.compare[ '*' ] );
			}

			toggleDiffTypeVisibility( inlineToggleSwitch.getValue() );

			$wikitextDiffBody.last().after( isInline ? $wikitextDiffBodyInline : $wikitextDiffBodyTable );
			$wikitextDiffBody = $wikitextDiffContainer.find( 'tr' ).not( $wikitextDiffHeader );
			/**
			 * Fired when the wikitext DOM is updated so others can react accordingly.
			 *
			 * @event ~'wikipage.diff.wikitextDiffBody'
			 * @memberof Hooks
			 * @param {jQuery} $wikitextDiffBody
			 */
			mw.hook( 'wikipage.diff.wikitextBodyUpdate' ).fire( $wikitextDiffBody );
		} );
	}

	/**
	 * Fired when the diff type switch is present so others can decide
	 * how to manipulate the DOM.
	 *
	 * @event ~'wikipage.diff.diffTypeSwitch'
	 * @memberof Hooks
	 * @param {OO.ui.ToggleSwitchWidget} inlineToggleSwitch
	 */
	mw.hook( 'wikipage.diff.diffTypeSwitch' ).fire( inlineToggleSwitch );
};